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内 容 简 介 


本 书 根据 作者 多 年 的 实际 教学 经 验 , 在 内 容 选择 、 理 论 深度 等 方面 进行 了 深入 的 分 析 和 研讨 ,使 学 生 
易于 理解 ,注重 对 学 生 的 启发 。 在 本 书 编写 过 程 中 ,力求 做 到 准确 性 、 系 统 性 .通俗 性 .实用 性 ,结构 清晰 ， 
注重 基础 理论 的 阐述 ,强调 理论 与 实践 的 结合 。 每 一 章 的 内 容 从 一 个 问题 开始 ,让 学 生 带 着 问题 开始 知识 
的 学 习 , 促 进 学 生 的 思考 和 参与 。 

全 书 共 分 为 9 章 , 主 要 内 容 包括 : 操作 系统 引 论 、 进 程 与 线程 .进程 并 发 控制 ,内存 管 理 ` 页 式 和 段 式 内 
存 管 理 ,1/O 管理 ,文件 管理 \ 死 锁 、 多 处 理 机 系统 介绍 。 

本 书 可 作为 高 等 院 校 计算 机 相关 专业 的 教材 ,也 可 供 从 事 计 算 机 工作 的 科技 人 员 进 行 参考 ,对 报考 研 
究 生 的 学 生 也 具有 一 定 的 参考 价值 。 
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出 版 说 明 


随 着 我 国 高 等 教育 规模 的 扩大 以 及 产业 结构 调整 的 进一步 完善 ,社会 对 高 层次 应 用 型 
人 才 的 需求 将 更 加 迫切。 各 地 高 校 紧 密 结 合 地 方 经 济 建设 发 展 需要 ,科学 运用 市 场 调节 机 
制 ,合理 调整 和 配置 教育 资源 ,在 改革 和 改造 传统 学 科 专 业 的 基础 上 ,加 强 工程 型 和 应 用 
学 科 专 业 建设 ,积极 设置 主要 面向 地 方 支柱 产业 、 高 新 技术 产业 、 服务 业 的 工程 型 和 应 用 型 
学 科 专 业 , 积 极为 地 方 经 济 建设 输送 各 类 应 用 型 人 才 。 各 高 校 加 大 了 使 用 信息 科学 等 现代 
科学 技术 提升 、 改 造 传统 学 科 专 业 的 力度 ,从 而 实现 传统 学 科 专 业 向 工程 型 和 应 用 型 学 科 专 
业 的 发 展 与 转变 。 在 发 挥 传统 学 科 专 业 师资 力量 强 、 办 学 经 验 丰 富 、 教 学 资源 充裕 等 优势 的 
同时 ,不 断 更 新 教学 内 容 、 改 革 课 程 体系 ,使 工程 型 和 应 用 型 学 科 专业 教育 与 经 济 建设 相 适 
应 。 计 算 机 课程 教学 在 从 传统 学 科 向 工程 型 和 应 用 型 学 科 转变 中 起 着 至 关 重 要 的 作用 , 工 
程 型 和 应 用 型 学 科 专 业 中 的 计算 机 课程 设置 .内容 体 系 和 教学 手段 及 方法 等 也 具有 不 同 于 
传统 学 科 的 鲜明 特点 。 

为 了 配合 高 校 工程 型 和 应 用 型 学 科 专业 的 建设 和 发 展 ,急需 出 版 一 批 内 容 新 、 体 系 新 、 
方法 新 、 手 段 新 的 高 水 平 计算 机 课程 教材 。 目 前 ,工程 型 和 应 用 型 学 科 专 业 计 算 机 课程 教材 
的 建设 工作 仍 滞后 于 教学 改革 的 实践 ,如 现 有 的 计算 机 教材 中 有 不 少 内 容 陈 旧 ( 依 然 用 传统 
专业 计算 机 教材 代替 工程 型 和 应 用 型 学 科 专 业 教材 ), 重 理论 、 轻 实践 ,不 能 满足 新 的 教学 计 
划 、 课 程 设置 的 需要 ; 一 些 课程 的 教材 可 供 选 择 的 品种 太 少 ; 一 些 基础 课 的 教材 虽然 品种 
较 多 ,但 低 水 平 重复 严重 ; 有 些 教材 内 容 庞 杂 , 书 越 编 越 厚 ; 专业 课 教 材 .教学 辅助 教材 及 
教学 参考 书 短缺 ,等 等 ,都 不 利于 学 生 能 力 的 提高 和 素质 的 培养 。 为 此 ,在 教育 部 相关 教学 
指导 委员 会 专家 的 指导 和 建议 下 ,清华 大 学 出 版 社 组 织 出 版 本 系列 教材 ,以 满足 工程 型 和 应 
型 学 科 专 业 计 算 机 课程 教学 的 需要 。 本 系列 教材 在 规划 过 程 中 体现 了 如 下 一 些 基 本 原则 
和 特点 。 

(1) 面向 工程 型 与 应 用 型 学 科 专 业 , 强 调 计算 机 在 各 专业 中 的 应 用 。 教 材 内 容 坚 持 基 
本 理论 适度 ,反映 基本 理论 和 原理 的 综合 应 用 ,强调 实践 和 应 用 环节 。 

(2) 反映 教学 需要 ,促进 教学 发 展 。 教 材 规划 以 新 的 工程 型 和 应 用 型 专业 目录 为 依据 。 
教材 要 适应 多 样 化 的 教学 需要 ,正确 把 握 教学 内 容 和 课程 体系 的 改革 方向 ,在 选择 教材 内 容 
和 编写 体系 时 注意 体现 素质 教育 ,创新 能 力 与 实践 能 力 的 培养 ,为 学 生 知识 、 能 力 、 素 质 协 调 
发 展 创造 条 件 。 

(3) 实施 精品 战略 ,突出 重点 ,保证 质量 。 规 划 教材 建设 仍然 把 重点 放 在 公共 基础 课 和 
专业 基础 课 的 教材 建设 上 ; 特别 注意 选择 并 安排 一 部 分 原来 基础 比较 好 的 优秀 教材 或 讲义 
修订 再 版 ,逐步 形成 精品 教材 ; 提倡 并 鼓励 编写 体现 工程 型 和 应 用 型 专业 教学 内 容 和 课程 
体系 改革 成 果 的 教材 。 
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握 作 系统 原理 


(4) 主张 一 网 多 本 ,合理 配套 。 基 础 课 和 专业 基础 课 教材 要 配套 ,同一 门 课程 可 以 有 多 
本 具有 不 同 内 容 特 点 的 教材 。 处 理 好 教材 统一 性 与 多 样 化 ,基本 教材 与 辅助 教材 ,教学 参考 
书 ,文字 教材 与 软件 教材 的 关系 ,实现 教材 系列 资源 配套 。 

(5) 依靠 专家 ,择优 选用 。 在 制订 教材 规划 时 要 依靠 各 课程 专家 在 调查 研究 本 课程 教 
材 建 设 现状 的 基础 上 提出 规划 选 题 。 在 落实 主编 人 选 时 ,要 引入 竞争 机 制 , 通 过 申报 、 评 审 
确定 主编 。 书 稿 完成 后 要 认真 实行 审 稿 程序 ,确保 出 书 质 量 。 

繁荣 教材 出 版 事业 ,提高 教材 质量 的 关键 是 教师 。 建 立 一 支 高 水 平 的 以 老 带 新 的 教材 
编写 队伍 才能 保证 教材 的 编写 质量 和 建设 力度 ,希望 有 志 于 教材 建设 的 教师 能 够 加 入 到 我 
们 的 编写 队伍 中 来 。 


21 世纪 高 等 学 校 计算 机 教育 实用 规划 教材 编 委 会 
联系 人 : 魏 江 江 weijj@tup. tsinghua. edu. cn 


前 言 


操作 系统 是 计算 机 系统 的 重要 组 成 部 分 ,是 用 户 使 用 计算 机 的 基础 。 作 为 计算 机 专业 
的 核心 课程 ,不 但 高 等 学 校 计算 机 相关 专业 的 学 生 必 须 学 习 , 从 事 计 算 机 行业 的 人 员 也 需要 
深入 了 解 。 但 是 很 多 学 生 在 学 习 的 过 程 中 都 觉得 操作 系统 这 门 课程 比较 抽象 枯燥 ,难以 理 
解 ,只 能 采取 死记 硬 背 的 方式 来 通过 考试 。 故 此 ,为 了 帮助 学 生 更 好 地 学 习 和 透彻 理解 计算 
机 系统 的 运行 过 程 和 操作 系统 的 基本 原理 ,一 种 适用 的 操作 系统 教材 显得 十 分 重要 。 

作者 在 多 年 的 教学 实践 和 科学 研究 的 基础 上 ,结合 操作 系统 教学 大 纲 、 研 究 生 入 学 考试 
要 求 和 全 国 计 算 机 技术 与 软件 专业 技术 资格 考试 大 纲 ,在 参考 了 国内 外 出 版 的 众多 操作 系 
统 教材 的 基础 上 编写 了 本 书 。 

1. 编写 背景 

国家 中 长 期 教育 改革 和 发 展 规划 纲要 (2010 一 2020) 指 出 : 注重 学 思 结 合 。 倡 导 启发 
式 、 探 究 式 、 讨 论 式 ,参与 式 教学 ,帮助 学 生 学 会 学 习 。 激 发 学 生 的 好 奇 心 ,培养 学 生 的 兴趣 
爱好 ,营造 独立 思考 、 自 由 探索 ,勇于 创新 的 良好 环境 。 适 应 经 济 社会 发 展 和 科技 进步 的 要 
求 , 推 进 课程 改革 ,加 强 教 材 建 设 ,建立 健全 教材 质量 监管 制度 。 

本 教材 就 是 按照 构建 创新 型 .应 用 型 人 才 培 养 模式 的 要 求 ,突出 对 学 生 实践 应 用 能 力 的 
培养 ,适应 社会 需求 。 从 问题 开始 ,按照 “提出 问题 "一 “分析 问 题 *>“ 明 确 目标 ”>“ 学 习 知 
识 ”>“ 解 决 问题 >“ 总结 提高 "的 思路 进行 内 容 组 织 。 激 发 学 生 学 习 的 主动 性 ,提高 学 生 的 
思考 能 力 和 创新 应 用 能 力 。 

2. 本 书 内 容 

全 书 共 分 为 9 章 , 主 要 内 容 如 下 。 

第 1 章 操作 系统 引 论 : 包括 计算 机 系统 与 操作 系统 ; 操作 系统 的 概念 ; 操作 系统 的 
历史 、 类 型 .功能 和 特性 ; 操作 系统 体系 结构 。 

第 2 章 进程 与 线程 : 包括 进程 的 概念 ,进程 控制 、 线 程 、 处 理 器 调度 。 

第 3 章 进程 并 发 控制 : 包括 并 发 概述 .PV 操作 、 进 程 同步 . 管 程 .进程 间 消 息 传递 。 

第 4 章 ”内 存 管理 : 包括 内 存 管理 概述 、 内 存 管理 的 基础 , 单 道 编程 中 的 内 存 管 理 、 多 
道 编程 中 的 内 存 管理 .空闲 空间 管理 。 

第 5 章 页 式 和 段 式 内 存 管理 : 包括 页 式 内存 管 理 、 页 面 更 新 算法 、 段 式 内 存 管理 、 虚 
拟 内 存 。 

第 6 章 I/O 管理: 包括 1/0 管理 概述 .IO 系统 .IO 缓冲 、 独 占 设备 的 分 配 、 设 备 处 
理 、 虚 拟 设备 、 磁 盘 管 理 、 磁 盘 高 速 缓存 .固态 盘 和 智能 磁盘 讨论 。 

第 7 章 文件 管理 : 包括 文件 管理 概述 .文件 组 织 和 存 取 、 目 录 管 理 ,文件 共享 与 安全 、 

辅 存 空间 管理 。 


所 作 系 统 原 理 


第 8 章 死 锁 : 包括 死 锁 原理 、 死 锁 检 测 、 死 锁 避 免 、 死 锁 预防 、 活 锁 与 饥饿 。 

第 9 章 多 处 理 机 系统 介绍 : 包括 多 处 理 机 基本 概念 ; 多 处 理 机 内 存 结构 ; 多 处 理 机 
操作 系统 类 型 ; 多 处 理 器 之 间 的 通信 ; 多 处 理 机 同步 ; 多 处 理 机 调度 ; 多 处 理 器 ; 超 线程 和 
多 核 的 比较 。 

3. 本 书 特 色 

(1) 充分 研讨 ,适合 教学 。 根 据 作 者 多 年 的 实际 教学 经 验 , 本 书 在 内 容 选择 、 理 论 深 度 
等 方面 进行 了 深入 的 分 析 和 研讨 ,使 本 书 易于 学 生理 解 ,尽量 满足 高 等 院 校 学 生 的 学 习 
需要 。 

(2) 由 浅 入 深 , 通 俗 易 懂 。 知 识 点 的 讲解 尽量 用 简洁 、 形 象 的 语言 来 表达 ,避免 过 于 元 
长 和 烦琐 的 表述 。 

(3) 问题 导入 ,以 问 开 始 。 每 一 章 的 内 容 从 一 个 问题 开始 ,让 学 生 带 着 问题 开始 知识 的 
学 习 , 促 进 学 生 的 思考 和 参与 ,在 知识 的 理解 中 去 解 开 对 问题 的 疑惑 。 

(4) 结构 清晰 ,注重 基础 。 整 体 知识 结构 清晰 明了 ,突出 对 基础 理论 的 阐述 ,注重 对 学 
生 的 启发 ,使 学 生 洞 彻 问题 的 核心 。 强 调理 论 与 实践 的 结合 ,让 学 生 在 实际 问题 的 探讨 中 充 
满 对 操作 系统 理论 的 神往 。 

(5) 配套 完善 ,满足 教学 。 提 供 对 应 的 PPT 课件 ,配套 出 版 的 (操作 系统 原理 习题 与 实 
验 指导 》 一 书 中 包括 : 例题 解析 、 课 后 自 测 题 、 自 测 题 答 案 及 分 析 、 实 验 指导 。 满 足 课堂 教 
学 、 课 后 练习 、 课 后 自 测 、 上 机 实验 的 一 体 化 需要 。 

本 书 第 3、4 章 由 于 世 东 编写 ,第 1.5 章 由 张丽娜 编写 ,第 2、8 章 由 董 丽 薇 编写 ,第 6.7 
章 由 穆 宝 良 编写 ,第 9 章 由 于 杨 编 写 。 高 源 副 教授 审阅 了 全 稿 并 提出 了 许多 有 益 的 意见 ; 
沈阳 工业 大 学 牛 连 强 教授 在 本 书 编写 过 程 中 给 予 了 指点 和 帮助 ,在 此 谨 向 他 们 表示 更 心 的 
感谢 。 感 谢 清华 大 学 出 版 社 在 本 书 的 出 版 过 程 中 给 予 的 支持 。 

由 于 作者 学 识 浅 陋 , 见 闻 不 广 , 书 中 必 有 不 足 之 处 , 敬 请 读者 提出 批评 、 指 正和 建议 。 我 
们 的 E-mail 地 址 是 : ysd0510@sina. com, 也 欢迎 大 家 与 我 们 进行 交流 和 探讨 。 


编 者 
2016 年 11 月 
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第 1 章 操作 系统 引 论 


操作 系统 (Operating System,OS) 是 计算 机 系统 中 最 重要 的 系统 软件 , 它 管理 整个 计算 
机 系统 的 软件 资源 和 硬件 资源 ,是 用 户 与 计算 机 硬件 的 桥梁 ,是 其 他 软件 和 程序 的 运行 基 
础 。 根 据 操作 系统 的 不 同 应 用 领域 ,各 种 操作 系统 有 着 不 同 的 设计 目标 和 设计 要 求 ,但 同 
时 ,它们 仍然 存在 着 共同 的 特征 。 

本 章 从 介绍 操作 系统 在 计算 机 系统 中 的 位 置 开 始 ,回顾 了 操作 系统 发 展 的 历史 ,介绍 了 
操作 系统 的 类 型 .功能 和 特征 ,并 对 支持 操作 系统 的 硬件 环境 及 操作 系统 设计 等 相关 问题 进 
行 综合 性 讨论 ,为 今后 进一步 学 习 操作 系统 理论 做 好 准备 。 


1.1 计算 机 系统 与 操作 系统 


1.1.1 计算 机 系统 的 组 成 


计算 机 系统 主要 由 硬件 资源 和 软件 资源 两 部 分 组 成 。 现 代 大 多 数 计算 机 系统 是 以 著名 
数学 家 冯 。 诺 依 曼 (Von Nueumann) 等 在 20 世纪 40 年 代 末 提出 的 “存储 程序 控制 ”的 原理 
为 基础 的 。 根 据 冯 。 诺 依 曼 的 分 析 , 计 算 机 必须 有 一 个 存储 器 用 来 存储 程序 和 数据 ; 有 一 
个 运算 器 用 于 执行 指定 的 操作 ; 有 一 个 控制 部 件 用 来 实现 操作 的 顺序 ; 还 要 有 输入 /输出 
设备 ,以 便 输入 数据 和 输出 计算 结果 。 因 此 ,硬件 资源 主要 包括 中 央 处 理 器 (CPU)、 存 储 
器 .输入 设备 和 输出 设备 。 只 由 硬件 设备 组 成 的 机 器 称 为 裸 机 。 

如 果 用 户 直 接 在 裸 机 上 处 理 程序 将 会 寸步 难 行 。 因 为 裸 机 不 包括 任何 软件 ,不 提供 任 
何 可 以 帮助 用 户 解 决 问 题 的 手段 ,没有 方便 应 用 程序 运行 的 环境 。 所 以 ,在 裸 机 上 必须 配置 
软件 ,以 满足 用 户 的 各 种 要 求 。 

软件 是 由 程序 、 数 据 和 在 研制 过 程 中 形成 的 各 种 文档 资料 组 成 ,是 方便 用 户 和 充分 发 挥 
计算 机 效能 的 各 种 程序 的 总 称 。 软 件 可 分 为 以 下 三 类 。 

(1) 系统 软件 : 操作 系统 、 编 译 程序 ,程序 设计 语言 ,以 及 与 计算 机 密切 相关 的 程序 。 

(2) 应 用 软件 : 各 种 应 用 程序 软件 包 。 

(3) 工具 软件 : 各 种 诊断 程序 检查 程序 、 引 导 程 序 。 

整个 计算 机 系统 的 组 成 可 用 图 1-1 来 描述 。 由 图 1-1 可 知 ,计算 机 系统 由 硬件 和 软件 
两 部 分 组 成 。 硬 件 处 于 计算 机 系统 的 底层 ; 软件 在 硬件 的 外 围 ,由 操作 系统 、 其 他 的 系统 软 
件 、 应 用 程序 构成 。 硬 件 是 计算 机 系统 的 物质 基础 ,没有 硬件 就 不 能 执行 指令 和 实施 最 基 
本 、 最 简单 的 操作 ,软件 也 就 失去 了 效用 ; 如 果 只 有 硬件 ,没有 配置 相应 的 软件 ,计算 机 也 不 


握 作 系统 原理 


其 他 系统 软件 


操作 系统 


用 户 程序 


图 1-1 计算 机 系统 的 组 成 


能 发 挥 它 的 潜在 能 力 , 这 些 硬件 资源 也 就 没有 了 活力 。 因 此 ,软件 和 硬件 有 机 地 结合 在 一 
起 ,构成 了 计算 机 系统 。 


1.1.2 OS 在 计算 机 系统 中 的 位 置 


在 计算 机 系统 中 ,操作 系统 的 位 置 处 在 硬件 和 其 他 所 有 软件 之 间 。 它 在 裸 机 上 运行 ,是 
所 有 软件 中 与 硬件 相连 的 第 一 层 软 件 。 从 操作 系统 在 计算 机 系统 中 的 位 置 可 以 分 析 操 作 系 
统 与 各 层 之 间 的 关系 ,这 对 于 理解 操作 系统 应 具备 的 功能 以 及 实现 这 些 功 能 的 方法 是 十 分 
重要 的 。 操 作 系统 与 各 层 的 关系 主要 表现 在 以 下 两 个 方面 。 

1. 操作 系统 对 各 层 的 管理 和 控制 

操作 系统 可 以 控制 CPU 的 工作 ,访问 存储 器 、 进 行 设备 驱动 和 设备 中 断 处 理 。 计 算 机 
用 户 可 以 通过 操作 系统 使 用 不 同 的 界面 ,方便 ,快捷 、 安 全 、 可 靠 地 操作 计算 机 硬件 来 完成 自 
己 的 计算 任务 。 

2. 计算 机 系统 各 层 对 操作 系统 的 制约 

1) 计算 机 系统 结构 对 操作 系统 实现 技术 的 制约 

硬件 提供 了 操作 系统 的 运行 基础 ,计算 机 的 系统 结构 对 操作 系统 的 实现 技术 有 着 重大 
的 影响 。 例 如 , 单 CPU 计算 机 的 特点 是 集中 顺序 过 程控 制 ,其 计算 模型 是 顺序 过 程 计算 模 
型 。 而 现代 操作 系统 大 多 数 是 多 用 户 、 多 任务 操作 系统 ,是 一 个 并 行 计算 模型 ,这 就 是 一 对 
矛盾 。 

单 CPU 计算 机 如 何 运 行 多 任务 呢 ? 为 此 ,操作 系统 提出 并 实现 了 以 下 各 章节 要 讨论 
的 内 容 , 使 得 在 单 CPU 的 计算 机 上 能 实现 多 任务 操作 系统 。 这 就 是 计算 机 的 系统 结构 对 
操作 系统 的 实现 技术 的 影响 和 制约 。 

2) 用 户 和 应 用 程序 的 需求 对 操作 系统 实现 技术 的 制约 

用 户 和 上 层 软 件 运行 在 操作 系统 提供 的 环境 上 ,对 操作 系统 会 提出 各 种 要 求 ,操作 系统 
必须 满足 不 同 的 应 用 需求 ,提供 良好 的 用 户 界 面 ,为 此 需要 设计 不 同类 型 的 操作 系统 。 


1.2 什么 是 操作 系统 


操作 系统 是 管理 和 控制 计算 机 硬件 与 软件 资源 的 计算 机 程序 ,是 直接 运行 在 " 裸 机 "上 
的 最 基本 的 系统 软件 ,任何 其 他 软件 都 必须 在 操作 系统 的 支持 下 才能 运行 。 


1.2.1 作为 用 户 与 计算 机 的 接口 


操作 系统 是 用 户 和 计算 机 的 接口 ,同时 也 是 计算 机 硬件 和 其 他 软件 的 接口 。 操 作 系 统 
是 计算 机 硬件 之 上 的 第 一 层 软 件 ,屏蔽 了 硬件 的 物理 特性 和 操作 细节 ,用 户 通 过 操作 系统 来 
使 用 计算 机 系统 。 用 户 在 操作 系统 的 帮助 下 能 够 方便 、 快 捷 、 可 靠 地 操纵 计算 机 硬件 和 运行 
自己 的 程序 。 


1.2.2 作为 系统 资源 的 管理 者 


有 效 地 管理 、 合 理 地 分 配 系统 资源 ,提高 系统 资源 的 使 用 效率 是 操作 系统 必须 发 挥 的 主 
要 作用 。 因 此 ,作为 系统 资源 的 管理 者 ,操作 系统 主要 完成 以 下 工作 。 

(1) 监视 各 种 资源 ,随时 记录 它们 的 状态 。 

(2) 实施 某 种 策略 以 决定 谁 获得 资源 , 何 时 获得 ,获得 多 少 。 

(3) 分 配 资源 供需 求 者 使 用 。 

(4) 回收 资源 ,以 便 再 次 分 配 。 


1.3 操作 系统 的 历史 


1.3.1 穿孔 卡片 


从 1946 年 第 一 台 计 算 机 诞生 至 20 世纪 50 年 代 中 期 ,一 直 未 出 现 操作 系统 ,计算 机 工 
作 采 用 手工 操作 方式 。 程 序 员 将 对 应 于 程序 和 数据 的 已 穿孔 的 卡片 (或 纸 带 ) 装 入 输入 机 ， 
然后 启动 输入 机 ,把 程序 和 数据 输入 计算 机 内 存 ; 通过 控制 台 开 关 启 动 程序 ,针对 数据 运 
行 ; 计算 完毕 后 ,打印 机 输出 计算 结果 ; 用 户 取 走 结果 并 务 下 卡片 (或 纸 带 ), 才 让 下 一 个 用 
户 使 用 计算 机 。 穿 孔 卡 片 和 穿孔 纸 带 如 图 1-2 和 图 1-3 所 示 。 


图 1-2 穿孔 卡片 图 1-3 穿孔 纸 带 
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穿孔 卡片 时 代 的 手工 操作 方式 具有 以 下 两 个 特点 。 

(1) 用 户 独 占 全 机 。 不 会 出 现 因 资 源 已 被 其 他 用 户 占用 而 等 待 的 现象 ,但 资源 的 利用 
率 低 。 

(2) CPU 等 待 手工 操作 ,CPU 的 利用 不 充分 。 

20 世纪 50 年 代 后 期 , 随 着 计算 机 运算 速度 的 加 快 ,人 机 矛盾 越 来 越 大 ,直至 无 法 容忍 。 
在 这 种 情况 下 ,必须 寻求 新 的 办 法 ,于 是 ,设计 并 实现 操作 系统 以 自动 完成 程序 的 装 入 和 运 
行 成 为 迫切 需要 。 这 样 ,就 出 现 了 批 处 理 系统 。 


1.3.2 简单 批 处 理 系统 


计算 机 发 展 的 早期 ,没有 任何 用 于 管理 的 软件 ,所 有 的 运行 管理 和 具体 操作 都 由 用 户 自 
己 承担 ,任何 操作 出 错 都 要 重 做 作业 ,CPU 的 利用 率 很 低 。 

为 此 ,解决 这 个 问题 的 方法 主要 有 两 个 : 一 个 是 配备 专门 的 计算 机 操作 员 , 程 序 员 不 再 
直接 操作 机 器 ,从 而 减少 操作 机 器 的 错误 ; 另 一 个 是 进行 批 处 理 , 操 作 员 把 用 户 提交 的 作业 
分 类 ,把 一 批 中 的 作业 编 成 一 个 作业 执行 序列 ,每 一 批 作业 将 有 专门 编制 的 监督 程序 
(Monitor) 自动 依次 处 理 。 当 一 批 作业 执行 完成 后 ,作业 又 把 控制 权 交 回 给 监督 程序 ,监督 
程序 再 将 磁带 上 的 第 二 批 作业 调 入 内 存 中 执行 ,以 此 类 推 ,直至 所 有 的 作业 都 完成 。 这 种 处 
理 方式 被 称 为 “ 批 处 理 方 式 ”。 早 期 批 处 理 的 操作 是 串 行 操作 ,所 以 被 称 为 简单 批 处 理 ,或 称 
为 单 道 批 处 理 。 

第 一 个 批 处 理 操作 系统 (也 是 第 一 个 操作 系统 ) 是 20 世纪 50 年 代 中 期 由 General 
Motors 开发 的 ,使 用 在 IBM 701 上 。 在 20 世纪 60 年 代 早期 ,许多 厂商 为 自己 的 计算 机 系 
统 开发 了 批 处 理 操作 系统 ,其 中 ,最 为 著名 的 是 用 于 IBM 7090/7094 计算 机 的 操作 系统 
IBSYS, 它 对 其 他 操作 系统 有 着 广泛 的 影响 ,如 图 1-4 所 示 。 
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图 1-4 简单 批 处 理 系 统 示 意图 


在 早期 的 简单 批 处 理 系统 中 ,作业 的 输入 和 输出 都 是 联机 的 。 联 机 I/O 的 缺点 是 速度 
慢 ,IO 设备 和 CPU 仍然 是 串 行 工作 ,CPU 利用 率 低 , 为 此 ,在 批 处 理 系 统 中 引入 了 脱 机 
WO 技术 。 除 主机 外 , 另 设 一 台 外 围 计算 机 ,该 机 仅 与 1/O 设备 交互 ,不 与 主机 相连 。 输 入 
设备 上 的 作业 通过 外 围 计算 机 输入 到 高 速 磁盘 上 ( 脱 机 输入 ) ,主机 从 高 速 磁盘 将 结果 读 出 


并 交 打 印 机 进行 打印 输出 。 这 样 ,I/O 工作 脱离 了 主机 ,外 围 计算 机 和 主机 可 以 并 行 工 作 ， 
加 快 了 程序 的 处 理 和 数据 的 输入 /输出 ,这 种 技术 称 为 脱 机 I/O 技术 ,如 图 1-5 所 示 。 
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图 1-5 脱 机 JIVO 技术 


1.3.3 多 道 批 处 理 系统 


在 简单 批 处 理 系统 中 ,内 存 中 仅 有 一 个 任务 ,无 法 充分 利用 系统 中 的 所 有 资源 ,致使 系 
统 中 仍 有 许多 资源 空闲 ,设备 利用 率 低 ,系统 性 能 差 。 在 20 世纪 60 年 代 中 期 ,计算 机 的 体 
系 结构 发 生 了 很 大 的 变化 ,由 以 CPU 为 中 心 的 机 构 改 变 为 以 主 存 为 中 心 ,使 在 内 存 中 同时 
装 入 多 个 作业 成 为 可 能 ,多 道 程序 的 概念 成 为 现实 。 

1. 多 道 程 序 设计 

多 道 程 序 设 计 技 术 是 指 允 许多 个 程序 同时 进入 内 存 并 运行 。 即 同时 把 多 个 程序 装 入 内 
存 , 并 允许 它们 交替 在 CPU 中 运行 ,它们 共享 系统 中 的 各 种 硬件 资源 和 软件 资源 。 当 一 道 
程序 因 1/O 请 求 而 暂停 运行 时 ,CPU 便 立 即 转 去 运行 另 一 道 程序 。 

多 道 程 序 合理 搭配 以 输入 /输出 为 主 和 以 计算 为 主 的 程序 ,使 得 它们 交 蔡 运行 ,从 而 充 
分 利用 资源 ,提高 系统 效率 。 

多 道 程序 的 运行 特点 是 计算 机 内 存 中 同时 存放 多 道 相 互 独立 的 程序 。 它 们 宏观 上 并 行 
运行 , 即 同 时 进入 系统 的 几 道 程序 都 处 于 运行 状态 ,但 都 未 运行 完成 ; 而 在 微观 上 是 串 行 运 
行 , 即 各 个 作业 轮流 使 用 CPU, 交 蔡 执 行 。 

2. 多 道 批 处 理 系统 

20 世纪 60 年 代 中 期 ,在 简单 批 处 理 系 统 中 ,引入 多 道 程序 设计 技术 后 形成 了 多 道 批 处 
理 系统 (简称 批 处 理 系统 ) 。 多 道 批 处 理 系统 的 特点 如 下 。 

1) 多 道 

系统 内 可 同时 容纳 多 个 作业 。 这 些 作业 存放 在 外 存 中 ,组 成 一 个 后 备 队列 ,系统 按 一 定 
的 调度 原则 每 次 从 后 备 作业 队列 中 选取 一 个 或 多 个 作业 进入 内 存 运行 ,作业 的 调度 由 系统 
自动 实现 ,从 而 在 系统 中 形成 一 个 自动 转 接 的 、 连 续 的 作业 流 。 

2) 成 批 

在 系统 运行 过 程 中 ,不 允许 用 户 与 其 作业 发 生 交互 作用 , 即 作 业 一 旦 进入 系统 ,用 户 就 
不 能 直接 干预 其 作业 的 运行 。 

多 道 批 处 理 系 统 的 主要 特征 有 以 下 三 个 方面 。 

(1) 用 户 脱 机 使 用 计算 机 : 作业 提交 后 直到 获得 结果 之 前 ,用 户 无 法 与 作业 交互 。 


操作 系统 引 论 


好 一 避 


妮 作 系统 原理 


(2) 作业 成 批 处 理 : 采用 成 批 处 理 作 业 。 

(3) 多 道 程 序 并 行 : 多 道 程序 在 内 存 中 并 行 运行 ,充分 利用 系统 资源 。 

多 道 批 处 理 系统 的 缺点 是 无 交互 性 ,用 户 一 旦 提交 作业 就 失去 了 对 其 运行 的 控制 能 力 ; 
同时 ,由 于 是 批 处 理 ,所 以 作业 的 周转 时 间 长 ,用 户 使 用 不 方便 。 


1.3.4 分 时 系统 


在 批 处 理 系 统 中 ,用 户 不 能 干预 自己 程序 的 运行 ,无 法 得 知 程序 的 运行 情况 ,这 对 程序 
的 调用 和 排 错 极为 不 利 。 为 了 克服 这 一 缺陷 ,增强 系统 的 交互 能 力 ,产生 了 分 时 操作 系统 
(Time Sharing Operating System) 。 

分 时 操作 系统 的 实现 思想 是 : 在 一 台 主 机 上 连接 多 个 带 有 显示 器 和 键盘 的 终端 ,同时 ， 
允许 多 个 用 户 通过 自己 的 终端 ,以 交互 方式 使 用 计算 机 ,共享 主机 资源 ,如 图 1-6 所 示 。 

分 时 技术 把 处 理 器 的 时 间 分 成 很 短 的 时 间 片 ,这 些 

时 间 片 轮流 地 分 配给 各 个 联机 的 作业 使 用 。 如 果 某 作业 

时 间 片 在 分 配给 它 的 时 间 片 用 完 时 仍 未 完成 , 则 该 作业 和 暂时 中 

断 , 等 待 下 一 轮 运 行 ,并 把 处 理 器 的 控制 权 让 给 另 一 个 作 

_ 业 使 用 。 这 样 在 一 个 相对 较 短 的 时 间 间 隔 内 ,每 个 用 户 
EE 时 作业 都 能 得 到 快速 响应 ,以 实现 人 机 交互。 

终端 终端 2 终 册 3 终端 第 一 个 分 时 操作 系统 是 由 麻 省 理工 学 院 开发 的 兼容 

分 时 系统 (Compatible Time-Sharing System,CTSS) , 源 

于 多 路 存 取 计算 项 目 , 该 系统 最 初 是 在 1961 年 为 

IBM 709 开发 的 ,后 来 又 移植 到 IBM 7094 中 。 

多 道 批 处 理 系统 和 分 时 系统 都 使 用 了 多 道 程 序 设 计 , 但 分 时 系统 与 多 道 批 处 理 系统 相 
比 ,具有 完全 不 同 的 特征 。 分 时 系统 具有 以 下 4 个 特点 。 

(1) 多 路 性 ; 允许 在 一 台 主 机 上 同时 连接 多 台 联 机 终端 ,系统 按 分 时 原则 为 每 个 用 户 
服务 。 

(2) 独立 性 ; 每 个 用 户 各 占 一 个 终端 ,彼此 独立 操作 , 互 不 干扰 。 

(3) 及 时 性 ; 用 户 的 请 求 能 在 很 短 的 时 间 内 获得 响应 。 

(4) 交互 性 : 用 户 可 通过 终端 与 系统 进行 广泛 的 人 机 对 话 。 


1.3.5 实时 系统 


虽然 多 道 批 处 理 操作 系统 和 分 时 操作 系统 获得 了 较 好 的 资源 利用 率 和 快速 的 响应 时 
间 , 从 而 使 计算 机 的 应 用 范围 日 益 扩大 ,但 它们 难以 满足 实时 控制 和 实时 信息 处 理 领 域 的 需 
求 。 这 样 就 产生 了 实时 系统 。 

目前 ,实时 系统 主要 包括 以 下 三 种 。 

(1) 过 程控 制 系统 。 计 算 机 用 于 生产 过 程 时 ,要 求 系统 能 现场 实时 采集 数据 ,并 对 采集 
的 数据 进行 及 时 处 理 , 进 而 能 自动 地 发 出 控制 信号 控制 相应 的 执行 机 构 , 使 某 些 参数 能 按 给 
定 的 规律 变化 ,以 保证 产品 质量 。 例 如 ,导弹 制导 系统 .飞机 自动 驾驶 系统 .火炮 自动 控制 系 
统 都 是 实时 过 程控 制 系统 。 

(2) 信息 查询 系统 。 情 报 检索 系统 是 典型 的 实时 信息 查询 系统 。 计 算 机 接收 成 百 上 干 


响应 时 
向 应 时 间 


123 


图 1-6 分 时 系统 示意 图 


从 各 个 终端 发 来 的 服务 请 求 和 提问 ,系统 应 在 极 快 的 时 间 内 做 出 回答 和 响应 。 

(3) 事务 处 理 系统 。 该 系统 不 但 对 终端 用 户 及 时 做 出 响应 ,而 且 要 对 系统 中 的 文件 或 
数据 库 频繁 更 新 。 例 如 ,银行 业务 处 理 系 统 ,每 次 银行 客户 发 生 业 务 往来 , 均 需 要 修改 相应 
的 文件 或 数据 库 。 这 样 的 系统 要 求 响应 快 . 安 全 保密 、 可 靠 性 高 。 

实时 操作 系统 (Real Time Operating System) 是 指 当 外 界 事件 或 数据 产生 时 ,能 够 接收 
并 以 足够 快 的 速度 予以 处 理 , 其 处 理 的 结果 又 能 在 规定 的 时 间 之 内 控制 监控 的 生产 过 程 或 
对 处 理 系统 做 出 快速 响应 ,并 控制 所 有 实时 任务 协调 一 致 运行 的 操作 系统 。 

实时 操作 系统 有 硬 实时 和 软 实时 之 分 , 硬 实时 要 求 在 规定 的 时 间 内 必须 完成 操作 ,这 是 
在 操作 系统 设计 时 保证 的 ; 软 实时 则 只 要 按照 任务 的 优先 级 , 尽 可 能 快 地 完成 操作 即 可 。 
我 们 通常 使 用 的 操作 系统 在 经 过 一 定 改变 之 后 就 可 以 变 成 实时 操作 系统 。 

分 时 系统 和 实时 系统 的 出 现 标志 着 操作 系统 步 入 了 实用 化 阶段 ,操作 系统 成 为 计算 机 
系统 中 重要 的 系统 软件 , 它 为 用 户 的 应 用 提供 了 一 个 良好 的 支撑 环境 ,方便 了 用 户 的 使 用 。 
批 处 理 操作 系统 、 分 时 操作 系统 和 实时 操作 系统 构成 了 现代 操作 系统 的 基本 类 型 ,现代 操作 
系统 可 能 综合 它们 多 方面 的 特征 以 满足 不 同 的 应 用 需求 。 


1.4 操作 系统 的 类 型 


1.4.1 大 型 计算 机 操作 系统 


大 型 计算 机 (Mainframe Computer) ,也 称 为 大 型 主机 。 大 型 计算 机 使 用 专用 的 处 理 器 
指令 集 、 操 作 系 统 和 应 用 软件 。 最 早 的 操作 系统 是 针对 20 世纪 60 年 代 的 大 型 主 结构 开发 
的 ,由 于 对 这 些 系统 在 软件 方面 做 了 巨大 投资 ,因此 ,原来 的 计算 机 厂商 继续 开发 与 原来 操 
作 系 统 相 兼 容 的 硬件 与 操作 系统 。 这 些 早期 的 操作 系统 是 现代 操作 系统 的 先驱 。 现 代 的 大 
型 主机 一 般 也 可 运行 Linux 或 UNIX 变种 。 


1.4.2 服务 器 操作 系统 


服务 器 操作 系统 (Server Operating System,SOS) ,又 称 为 网 络 操作 系统 ,一 般 指 的 是 安 
装 在 大 型 计算 机 上 的 操作 系统 ,例如 Web 服务 器 、 应 用 服务 器 和 数据 库 服务 器 等 ,是 企业 
IT 系统 的 基础 架构 平台 。 

同时 ,服务 器 操作 系统 也 可 以 安装 在 个 人 计算 机 上 。 相 比 个 人 版 操作 系统 ,在 一 个 具体 
的 网 络 中 ,服务 器 操作 系统 要 承担 额外 的 管理 ,配置 .稳定 、 安 全 等 功能 ,处 于 每 个 网 络 中 的 
心脏 部 位 。 

服务 器 操作 系统 主要 分 为 4 大 流派 : Windows、NetWare、UNIX、Linux。 

Windows 服务 器 操作 系统 是 全 球 最 大 的 操作 系统 开发 商 一 一 Microsoft 公司 开发 的 。 
其 服务 器 操作 系统 重要 版 本 有 Windows NT 4. 0 Server、Windows NT Server、 Windows 
Server 2003、Windows Server 2008、Windows Server 2008 R2、Windows Server 2012、 
Windows Server Technical 等 。 

现在 一 些 IT 圈 里 的 朋友 对 NetWare 服务 器 操作 系统 可 能 比较 陌生 ,由 于 种 种 原因 ， 
的 市 场 占 有 率 已 经 非常 局 限 ,主要 应 用 在 某 些 特定 的 行业 中 。 目 前 ,在 一 些 特定 行 业 和 事业 
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单位 中 , NetWare 优秀 的 批 处 理 功能 和 安全 、 稳 定 的 系统 性 能 还 有 很 大 的 生存 空间 。 
NetWare 目前 常用 的 版 本 主要 有 Novell 的 3. 11.3. 12 .4. 10 .5.0 等 中 英文 版 。 

UNIX 服务 器 操作 系统 由 AT&T 公司 和 SCO 公司 共同 推出 ,主要 支持 大 型 的 文件 系 
统 服务 .数据 服务 等 应 用 。 由 于 一 些 出 众 的 服务 器 厂商 生产 的 高 端 服务 器 产品 中 甚至 只 支 
持 UNIX 操作 系统 ,因而 在 很 多 人 的 眼中 ,UNIX 甚至 成 为 高 端 操作 系统 的 代名词 。 目 前 市 
上 流行 的 主要 有 Sun Solaris,IBM-AIX、HP-UX、FreeBSD、OS X Server 等 。 

Linux 服务 器 操作 系统 是 国外 几 位 IT 前 辈 , 在 POSIX 和 UNIX 基础 上 开发 出 来 的 , 支 
特 多 用 户 、 多 任务 、 多 线程 .多 CPU。Linux 开放 源 代码 政策 ,使 得 基于 其 平台 的 开发 与 使 
日 无 须 支 付 任何 单位 和 个 人 的 版 权 费 用 ,成 为 后 来 很 多 操作 系统 厂家 创业 的 基石 ,同时 也 成 
为 目前 国内 外 很 多 保密 机 构 服 务 器 操作 系统 采购 的 首选 。 目 前 ,国内 主流 市 场 中 使 用 的 主 
要 有 Novell 的 中 文 版 Suse Linux 9.0、 小红帽 系列 .红旗 Linux 系列 等 。 


1.4.3 个 人 计算 机 操作 系统 


随 着 计算 机 应 用 的 日 益 广泛 ,许多 人 都 能 拥有 自己 的 个 人 计算 机 ,而 在 大 学 、 政 府 部 门 
或 商业 系统 中 则 使 用 功能 更 强 的 个 人 计算 机 ,通常 称 为 工作 站 。 在 个 人 计算 机 上 配置 的 操 
作 系 统称 为 个 人 计算 机 操作 系统 。 

目前 ,在 个 人 计算 机 和 工作 站 领域 有 两 种 主流 操作 系统 : 一 种 是 微软 公司 提供 的 具有 
图 形 用 户 界 面 的 视窗 操作 系统 Windows; 另 一 种 是 UNIX 系统 和 Linux 系统 。 

Windows 系统 的 前 身 是 MS-DOS。MS-DOS 是 微软 公司 早期 开发 的 磁盘 操作 系统 ,其 
应 用 十 分 广泛 ,具有 设备 管理 ,文件 系统 功能 ,提供 键盘 命令 和 系统 调用 命令 。 后 来 , MS- 
DOS 逐渐 发 展 成 为 界面 色彩 丰富 、 使 用 直观 方便 .具有 图 形 用 户 界面 的 Windows 操作 

UNIX 系统 是 一 个 多 用 户 分 时 操作 系统 , 自 1970 年 问世 以 来 十 分 流行 , 它 运行 在 从 
高 档 个 人 计算 机 到 大 型 计算 机 等 各 种 不 同 处 理 能 力 的 机 器 上 ,提供 了 良好 的 工作 环境 ; 
它 具 有 可 移植 性 、 安 全 性 ,提供 了 很 好 的 网 络 支持 功能 ,大 量 用 于 网 络 服务 器 。 而 目前 十 
分 受 欢迎 的 \. 开 放 源 码 的 操作 系统 Linux, 则 是 用 于 个 人 计算 机 的 、 类 似 UNIX 的 操作 


1.4.4 多 处 理 机 操作 系统 


广义 上 说 ,使 用 多 台 计 算 机 协同 工作 来 完成 所 要 求 的 任务 的 计算 机 系统 都 是 多 处 理 机 
系统 。 传 统 的 狭义 多 处 理 机 系统 是 指 利用 系统 内 的 多 个 CPU 并 行 执行 用 户 多 个 程序 ,以 
提高 系统 的 吞吐 量 或 用 来 进行 元 余 操 作 以 提高 系统 的 可 靠 性 。 

多 处 理 机 系统 是 多 个 处 理 机 (器 ) 在 物理 位 置 上 处 于 同一 机 壳 中 ,有 一 个 单一 的 系统 物 
理 地 址 空间 ,每 一 个 处 理 机 均 可 访问 系统 内 的 所 有 存储 器 。 

多 处 理 机 操作 系统 (Multiprocessors Operating System) 一 般 应 用 于 并 行 处 理 机 。 并 行 
处 理 机 又 叫 SIMD 计算 机 。 它 是 单一 控制 部 件 控制 下 的 多 个 处 理 单元 构成 的 阵列 ,所 以 又 
称 为 阵列 处 理 机 。 多 处 理 机 是 由 多 台独 立 的 处 理 机 组 成 的 系统 。 

多 处 理 机 操作 系统 ,目前 有 以 下 三 种 类 型 。 


1, 主 从 式 
主 从 式 (Master-Slave) 操 作 系 统 由 一 台 主 处 理 机 记录 、 控 制 其 他 从 处 理 机 的 状态 ,并 分 
配 任务 给 从 处 理 机 。 

2. 独立 监督 式 

独立 监督 式 (Separate Supervisor) 与 主 从 式 不 同 ,在 这 种 类 型 中 ,每 一 个 处 理 机 均 有 各 
自 的 管理 程序 (核心 )。 采 用 独立 监督 式 操作 系统 的 多 处 理 机 系统 有 IBM 370/158 等 。 

3. 浮动 监督 式 

浮动 监督 式 (Floating Supervisor) 中 每 次 只 有 一 台 处 理 机 作为 执行 全 面 管 理 功能 的 “ 主 
处 理 机 ”, 但 根据 需要 ,“ 主 处 理 机 ”是 可 浮动 的 , 即 从 一 台 切 换 到 另 一 台 处 理 机 。 这 是 最 复 
杂 、 最 有 效 、 最 灵活 的 一 种 多 处 理 机 操作 系统 ,常用 于 对 称 多 处 理 机 系统 ( 即 系 统 中 所 有 处 理 
机 的 权限 是 相同 的 ,有 公用 主 存 和 I/O 子 系统 ) 。 

多 处 理 机 操作 系统 的 优点 是 : 允许 多 个 进程 同时 运行 在 多 个 处 理 机 上 ,对 于 大 型 计算 
任务 ,相对 单 处 理 机 ,性 能 有 较 大 的 提升 。 缺 点 是 : 处 理 器 的 数量 不 可 以 随意 增加 , 即 计 算 
能 力 有 上 限 。 


1.4.5 移动 设备 操作 系统 


移动 设备 操作 系统 (Mobile Operating System ,MOS) 主 要 应 用 在 智能 手机 上 。 主 流 的 
智能 手机 移动 设备 操作 系统 有 Google Android 和 苹果 的 iOS 等 。 智 能 手机 与 非 智 能 手机 
都 支持 Java, 智能 手机 与 非 智 能 手机 的 区 别 主要 看 其 能 否 基 于 系统 平台 的 功能 扩展 , 非 
Java 应 用 平台 ,还 有 就 是 支持 多 任务 。 

移动 设备 操作 系统 一 般 应 用 在 智能 手机 上 。 目 前 ,在 智能 手机 市 场 上 仍 以 个 人 信息 管 
理 型 手机 为 主 , 随 着 更 多 厂商 的 加 入 ,整体 市 场 的 竞争 已 经 开始 呈现 出 分 散 化 的 态势 。 从 市 
场 容 量 .竞争 状态 和 应 用 状况 上 来 看 ,整个 市 场 仍 处 于 启动 阶段 。 目 前 应 用 在 手机 上 的 操作 
系统 主要 有 Android (谷歌 )、iOS (苹果 )、Windows Phone( 微 软 )、Symbian (诺基亚 )、 
BlackBerry OS( 黑 莓 )\Windows Mobile( 微 软 ) 等 。 


1.4.6 和 骸 入 式 操 作 系统 


嵌入 式 操作 系统 (Embedded Operating System,EOS) 是 一 种 用 途 广泛 的 系统 软件 ,过 
去 它 主 要 应 用 于 工业 控制 和 国防 系统 领域 。EOS 负责 嵌入 系统 的 全 部 软件 和 硬件 资源 的 
分 配 及 任务 调度 .控制 .协调 并 发 活动 。 它 必须 体现 其 所 在 系统 的 特征 ,能 够 通过 装卸 某 些 
模块 来 达到 系统 所 要 求 的 功能 。 

某 些 情 况 下 ,嵌入 式 操作 系统 指 的 是 一 个 自 带 了 固定 应 用 软件 的 巨大 泛 用 程序 。 在 许 
多 最 简单 的 嵌入 式 系统 中 ,所 谓 的 操作 系统 就 是 指 其 上 唯一 的 应 用 程序 。 

流行 的 嵌入 式 操作 系统 包括 VxWorks、Nucleus、Windows CE .嵌入 式 Linux 等 ,它们 
广泛 应 用 于 国防 系统 、 工 业 控制 、 交 通 管理 ,信息 家 电 、 家 庭 智 能 管理 .POS 网 络 、 环 境 工 程 
与 自然 监测 、 机 器 人 等 多 种 领域 。 


1.4.7 智能 卡 操作 系统 
智能 卡 操作 系统 (Chip Operating System.COS) ,一般 是 紧 紧 围绕 着 它 所 服务 的 智能 卡 
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的 特点 而 开发 的 。 由 于 不 可 避免 地 受到 了 智能 卡 内 微 处 理 器 芯片 的 性 能 及 内 存 容量 的 影 
响 , 因 此 ,COS 在 很 大 程度 上 不 同 于 人 们 通常 所 能 见 到 的 微机 上 的 操作 系统 (例如 DOS、 
UNIX 等 ) 。 

首先 ,COS 是 一 个 专用 系统 而 不 是 通用 系统 。 即 一 种 COS 一 般 都 只 能 应 用 于 特定 的 
某 种 (或 者 是 某 些 ) 智能卡, 不 同 卡 内 的 COS 一 般 是 不 相同 的 。 因 为 COS 一 般 都 是 根据 某 
种 智能 卡 的 特点 及 其 应 用 范围 而 特定 设计 开发 的 ,尽管 它们 在 所 实际 完成 的 功能 上 可 能 
部 分 都 遵循 着 同一 个 国际 标准 。 

其 次 ,与 那些 常见 的 微机 上 的 操作 系统 相 比较 而 言 ,COS 在 本 质 上 更 加 接近 于 临 控 程 
序 ,而 不 是 一 个 通常 所 谓 的 真正 意义 上 的 操作 系统 ,这 一 点 至 少 在 目前 看 来 仍 是 如 此 。 因 为 
在 当前 阶段 ,COS 所 需要 解决 的 主要 还 是 对 外 部 的 命令 如 何 进行 处 理 、 响 应 的 问题 ,这 其 中 
一 般 并 不 涉及 共享 .并 发 的 管理 及 处 理 ,而 且 就 智能 卡 在 目前 的 应 用 情况 而 看 ,并 发 和 共享 
的 工作 也 确实 是 不 需要 的 。 

COS 在 设计 时 一 般 都 是 紧密 结合 智能 卡 内 存储 器 分 区 的 情况 ,按照 国际 标准 (ISO/ 
IEC 7816 系列 标准 ) 中 所 规定 的 一 些 功 能 进行 设计 .开发 。 但 是 由 于 目前 智能 卡 的 发 展 速 
度 很 快 ,而 国际 标准 的 制定 周期 相对 比较 长 一 些 ,因而 造成 了 当前 的 智能 际 标 准 还 不 太 
完善 的 情况 , 据 此 ,许多 厂家 又 各 自 都 对 自己 开发 的 COS 做 了 一 定 的 扩充 。 

传统 的 COS 和 卡片 应 用 是 在 安全 的 环境 下 开发 并 装载 到 芯片 内 的 ,最 近 几 年 ,开放 式 
操作 系统 平台 如 Java CardTM、MultOS、Windows For Smart Card 取得 了 重大 发 展 , 这 大 
大 方便 了 智能 IC 卡 的 应 用 开发 和 一 卡 多 用 的 实现 .并且 允许 动态 地 装载 .更 新 或 删除 卡片 
应 用 。 

微软 智能 IC 卡 视窗 (Windows For Smart Card) 与 微软 Windows 操作 系统 相 结合 ,将 
在 电子 商务 、 网 络 安全 方面 有 广阔 的 前 景 。MULTOS 是 一 个 多 应 用 OS, 它 的 卡片 在 有 效 
生命 周期 内 允许 动态 地 装载 ,更 新 或 删除 卡片 应 用 。 
另外 ,智能 IC 卡 也 是 电子 商务 的 未 来 , 它 本 身 固有 的 安全 性 和 方便 性 ,使 其 成 为 目前 公 
认 的 网 络 安全 用 户 端 解 决 方案 。 利 用 智能 IC 卡 可 以 较 方便 地 通过 数据 加 密 以 及 通过 PKI 
进行 身份 验证 ,保证 在 线 安全 支付 。 


1.5 操作 系统 的 功能 和 特征 


1.5.1 操作 系统 的 功能 


操作 系统 是 管理 和 控制 计算 机 系统 中 的 所 有 硬件 、 软 件 资源 ,合理 地 组 织 计 算 机 工作 流 
程 , 并 为 用 户 提供 一 个 良好 的 工作 环境 和 友好 的 接口 。 计 算 机 系统 的 主要 硬件 资源 有 处 理 
器 存储器、 外 部 设备 ,软件 资源 以 文件 形式 存在 外 存储 器 上 。 因 此 从 资源 管理 和 用 户 接口 
的 观点 来 看 ,操作 系统 具有 处 理 机 管理 .存储 管理 .设备 管理 文件 系统 管理 和 用 户 接口 管理 
5 种 功能 。 

1. 处 理 机 管理 

计算 机 系统 中 最 重要 的 资源 是 中 央 处 理 机 (Central Processing Unit,CPU) ,任何 计算 
都 必须 在 CPU 上 进行 。 在 处 理 机 管理 中 ,最 核心 的 问题 是 CPU 时 间 的 分 配 问题 ,这 涉及 


分 配 的 策略 和 方法 。 在 单 CPU 计算 机 系统 中 , 当 有 多 进程 请 求 CPU 时 ,将 处 理 机 分 配给 
哪个 进程 使 用 的 问题 就 是 处 理 机 分 配 的 策略 问题 。 调 度 策略 也 是 分 配 原则 ,这 是 在 多 对 一 
的 情况 下 ( 即 多 个 进程 竞争 一 个 CPU) 必 须 确定 的 。 这 些 原 则 因 系 统 的 设计 目标 不 同 而 不 
同 。 可 以 按 进 程 的 紧迫 程度 ,或 按 进程 发 出 请 求 的 先后 次 序 ,或 是 其 他 的 原则 来 确定 处 理 机 
的 分 配 原则 。 

在 确定 调度 策略 时 ,还 需要 确定 给 定 的 CPU 时 间 , 是 分 配 一 个 时 间 片 ? 还 是 让 选中 进 
程 占用 CPU, 直 到 该 进程 因为 请 求 1/O 操作 等 原因 放弃 CPU 控制 权 ? 

最 后 ,还 需要 解决 的 问题 是 给 选中 的 进程 进行 处 理 机 的 分 配 ,使 选中 的 进程 真正 得 到 
CPU 的 控制 权 。 因 此 ,处 理 机 管理 的 功能 是 : 

(1) 确定 进程 调度 策略 ; 

(2) 给 出 进程 调度 算法 ; 

(3) 进行 处 理 机 的 分 配 。 

2. 存储 管理 

存储 管理 的 主要 工作 是 对 内 存储 器 进行 合理 分 配 、 有 效 保护 和 扩充 。 内 存 是 现代 计算 
机 系统 的 中 心 ,是 可 以 被 CPU 和 I/O 设备 共同 访问 的 数据 仓库 。 内 存 通 常 是 CPU 直接 寻 
址 和 访问 的 ,唯一 的 大 容量 存储 器 。 

为 了 改善 CPU 的 利用 率 和 计算 机 对 用 户 的 响应 速度 ,必须 在 内 存 中 保留 多 个 程序 。 
内 存 管 理 方法 很 多 ,不 同 算法 的 效能 和 特定 环境 有 关 。 某 一 特定 系统 的 内 存 管 理 方法 的 选 
择 取决 于 多 种 因素 ,尤其 是 系统 的 硬件 设计 。 内 存 管理 主要 管理 以 下 内 存活 动 。 

(1) 记录 内 存在 哪些 部 分 正在 被 使 用 及 被 谁 使 用 。 

(2) 当 内 存 空间 可 用 时 ,决定 哪些 进程 可 以 装 入 内 存 。 

(3) 根据 需要 分 配 和 释放 内 存 空间 。 

(4) 确保 在 多 道 程序 环境 下 ,各 个 程序 只 在 自己 的 内 存 空间 中 运行 , 互 不 干扰 。 

(5) 当 内 存 空间 不 足 时 ,采取 何 种 策略 扩展 逻辑 内 存 。 

3. 设备 管理 

设备 管理 是 操作 系统 中 最 庞杂 、 琐 碎 的 部 分 ,其 原因 是 : 

(1) 设备 管理 涉及 很 多 实际 的 物理 设备 ,这 些 设备 品种 繁多 、 用 法 各 异 。 

(2) 各 种 外 部 设备 都 能 和 主机 并 行 工作 ,而 且 , 有 些 设 备 可 被 多 个 进程 所 共享 。 

(3) 主机 和 外 部 设备 ,以 及 各 类 外 部 设备 之 间 的 速度 极 不 匹配 , 极 差 很 大 。 

基于 以 上 原因 ,现代 操作 系统 的 设备 管理 主要 解决 以 下 问题 。 

1) 设备 无 关 性 

用 户 向 系统 申请 和 使 用 的 设备 与 实际 操作 的 设备 无 关 , 即 在 用 户 程序 中 或 在 资源 申请 
命令 中 使 用 设备 的 逻辑 名 , 即 设备 无 关 性 。 这 一 特征 不 仅 为 用 户 使 用 设备 提供 了 方便 ,而 且 
也 提高 了 设备 的 利用 率 。 

2) 设备 分 配 

各 个 用 户 程序 在 其 运行 的 开始 阶段 .中 间或 结束 时 都 可 能 要 进行 输入 或 输出 ,因此 需要 
请 求 使 用 外 部 设备 。 在 一 般 情况 下 ,外 部 设备 的 种 类 与 台数 是 有 限 的 ,所 以 ,在 多 个 用 户 请 
求 少量 设备 台数 的 情况 下 ,如 何 分 配 设备 是 十 分 重要 的 。 设 备 分 配 通 常 采用 独 享 分 配 、 共 享 
分 配 和 虚拟 分 配 三 种 基本 分 配 技术 。 
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3) 设备 的 传输 控制 

设备 的 传输 控制 是 设备 管理 要 完成 的 重要 和 本 质 工作 。 主 要 工作 包括 : @ 控 制 设 备 实 
岗 物 理 I/O 操作 , 即 组 织 完 成 本 次 IO 操作 的 有 关 信 息 ,启动 设备 工作 ; 加 当 设 备 完成 本 次 
1/O 操作 或 操作 出 错时 会 产生 设备 中 断 信 号 ,由 设备 中 断 处 理 程序 进行 中 断 处 理 。 
另外 ,设备 管理 还 提供 缓冲 技术 , SPOOLing 技术 和 改造 设备 特性 和 提高 设备 的 利 


4. 文件 系统 管理 
以 上 三 种 管理 都 是 针对 计算 机 的 硬件 资源 的 管理 。 文 件 系统 管理 则 是 对 软件 资源 的 管 
理 。 为 了 管理 庞大 的 系统 软件 资源 及 用 户 提供 的 程序 和 数据 ,操作 系统 将 它们 组 织 成 文件 
的 形式 ,操作 系统 对 软件 的 管理 实际 上 是 对 文件 系统 的 管理 。 

文件 系统 要 解决 的 问题 是 ,为 用 户 提供 一 种 简便 的 、 统 一 的 存 取 和 管理 信息 的 方法 ,并 
要 解决 信息 的 共享 ,数据 的 存 取 控制 和 保密 等 问题 。 具 体 而 言 ,文件 系统 要 实现 用 户 的 信息 
组 织 、 提 供 存 取 方 法 、 实 现 文件 共享 和 文件 安全 ,还 要 保证 文件 完整 性 ,完成 磁盘 空间 分 配 的 
任务 。 

5. 用 户 接口 管理 

计算 机 用 户 与 计算 机 的 交流 是 通过 操作 系统 的 用 户 接口 (或 称 用 户 界面 ) 完 成 的 。 操 作 
系统 为 用 户 提供 的 接口 有 两 种 ,一 是 操作 界面 ; 二 是 操作 系统 的 功能 服务 界面 。 


1.5.2 操作 系统 的 特征 


操作 系统 的 基本 特征 有 4 个 ,分 别 是 并 发 性 .共享 性 .虚拟 性 和 异步 性 。 
1. 并 发 性 
行 性 与 并 发 (Concurrence) 性 这 两 个 概念 是 既 相 似 又 区 别 的 两 个 概念 。 并 行 性 是 指 
两 个 或 者 多 个 事件 在 同一 时 刻 发 生 , 这 是 一 个 具有 微观 意义 的 概念 , 即 在 物理 上 这 些 事 
件 是 同时 发 生 的 ; 而 并 发 性 是 指 两 个 或 者 多 个 事件 在 同一 时 间 间 隔 内 发 生 , 它 是 一 个 较 
为 宏观 的 概念 。 在 多 道 程序 环境 下 ,并 发 性 是 指 在 一 段 时 间 内 有 多 道 程序 在 同时 运行 ， 
但 在 单 处 理 机 的 系统 中 ,每 一 时 刻 仅 能 执行 一 道 程序 , 故 微观 上 这 些 程序 是 在 交替 执 
行 的 。 

通常 的 程序 是 静态 实体 ,它们 是 不 能 并 发 执行 的 。 为 了 使 程序 能 并 发 执行 ,系统 必须 分 
别 为 每 个 程序 建立 进程 。 进 程 ,又 称 任务 ,简单 来 说 ,是 指 在 系统 中 能 独立 运行 并 作为 资源 
分 配 的 基本 单位 , 它 是 一 个 活动 的 实体 。 多 个 进程 之 间 可 以 并 发 执行 和 交换 信息 。 一 个 进 
程 在 运行 时 需要 一 定 的 资源 ,如 CPU、 存 储 空间 及 1/O 设备 等 。 在 操作 系统 中 引入 进程 的 
目的 是 使 程序 能 并 发 执行 。 

2. 共享 性 

所 谓 共 享 (Sharing) 是 指 系 统 中 的 资源 可 供 内 存 中 多 个 并 发 执行 的 进程 共同 使 用 。 由 
于 资源 的 属性 不 同 , 故 多 个 进程 对 资源 的 共享 方式 也 不 同 ,可 以 分 为 : 互 斥 共享 方式 和 同时 
访问 方式 。 

3. 虚拟 性 

虚拟 (Virtual) 性 是 指 通过 技术 把 一 个 物理 实体 变 成 若干 个 逻辑 上 的 对 应 物 。 在 操作 
系统 中 虚拟 的 实现 主要 是 通过 分 时 的 使 用 方法 。 显 然 , 如 果 n 是 某 一 个 物理 设备 所 对 应 的 


虚拟 逻辑 设备 数 , 则 虚拟 设备 的 速度 必然 是 物理 设备 速度 的 1/n。 

4. 异步 性 

在 多 道 程序 设计 环境 下 ,允许 多 个 进程 并 发 执行 ,由 于 资源 等 因素 的 限制 ,通常 ,进程 的 
执行 并 非 “一 气 呵 成 ”, 而 是 以 “ 走 走 停 停 ”的 方式 运行 。 内 存 中 每 个 进程 在 何 时 执行 , 何 时 和 暂 
停 ,以 怎样 的 方式 向 前 推进 ,每 道 程序 总 共 需 要 多 少时 间 才 能 完成 ,都 是 不 可 预知 的 。 或 者 
说 ,进程 是 以 异步 (Asynchronism) 的 方式 运行 的 。 尽 管 如 此 ,但 只 要 运行 环境 相同 ,作业 经 
过 多 次 运行 ,都 会 获得 与 单 道 运行 时 完全 相同 的 结果 。 因 此 ,异步 性 是 并 发 性 的 表现 特征 ， 
并 发 性 是 异步 性 的 内 在 原因 。 


1.6 操作 系统 体系 结构 


在 操作 系统 的 发 展 过 程 中 ,产生 了 多 种 体系 结构 。 到 目前 为 止 ,主要 分 为 4 种 类 型 : 单 
体 结构 .层次 式 结构 .虚拟 机 结构 和 C/S 结构 。 


1.6.1 单 体 结构 


单 体 结构 又 称 为 模块 组 合 结构 ,是 一 种 基于 结构 化 程序 设计 的 软件 设计 方法 。 早 期 的 
操作 系统 (如 IBM S/360) 及 一 些小 型 操作 系统 (如 DOS) 都 属于 这 种 类 型 。 

单 体 结构 操作 系统 的 基本 设计 思想 是 : 把 模块 作为 操作 系统 的 基本 单位 ,按照 功能 需 
求 把 整个 操作 系统 分 解 成 若干 个 模块 ,每 个 模块 具有 一 定 的 功能 ,若干 个 关联 模块 协作 完成 
某 个 功能 。 各 个 模块 可 以 不 加 控制 .自由 调用 ,每 个 模块 经 独立 设计 、 编 码 和 调试 后 连接 完 
成 一 个 完整 的 系统 。 


单 体 结构 的 优点 是 : 程序 结构 紧密 ,接口 简 全 一 站 
单 直接 ,系统 效率 高 。 但 它 也 有 一 些 缺 点 ,如 模 


块 独立 性 差 ,模块 之 间 联 系 太 多 ,系统 结构 不 清 
晰 ,系统 的 正确 性 难以 保证 ,可 靠 性 降低 ,扩充 性 
差 等 ,会 “ 牵 一 发 而 动 全 身 ”, 如 图 1-7 所 示 。 


1.6.2 层次 式 结构 


为 了 保证 操作 系统 结构 的 清晰 ,具有 较 高 的 可 靠 性 和 较 强 的 适应 性 ,易于 扩充 和 移植 ， 
在 单 体 结构 操作 系统 的 基础 上 产生 了 层次 式 结构 的 操作 系统 。 

层次 式 结构 操作 系统 的 设计 思想 是 : 把 操作 系统 的 所 有 功能 模块 按照 功能 的 调用 次 
序 分 别 排 成 若干 层 ,各 层 之 间 的 模块 只 能 单 向 依赖 或 单 向 调用 , 即 只 允许 上 层 或 外 层 模 
块 调用 下 层 或 内 层 模 块 ,这 样 不 但 会 使 操作 系统 的 结构 清晰 ,而 且 不 构成 循环 ,如 图 1-8 
所 示 。 

层次 式 结构 操作 系统 的 经 典 案 例 是 1968 年 由 E. W. Dijkstra 和 他 的 学 生 们 建造 的 
THE 系统 ,该 系统 的 设计 目标 是 实现 一 个 可 证 明正 确 性 的 操作 系统 , 它 是 第 一 个 按 层次 
式 结 构 构 造 的 操作 系统 。 它 是 一 个 简单 的 批 处 理 系统 ,该 系统 有 6 层 , 其 层次 如 图 1-9 
所 示 。 


图 1-7 单 体 结构 示意 图 
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攻 一 避 


所 作 系 统 原 理 


TN 第 5 层 操作 员 
第 4 层 用 户 程序 
ee 第 3 层 输入 /输出 管理 
第 1 层 第 2 层 操作 员 一 进程 通信 
pe 第 1 层 内 存 和 磁 鼓 管理 
0B | 第 0 层 处 理 机 分 配 和 多 道 程序 环境 
图 1-8 层次 式 结构 示意 图 图 1-9 THE 操作 系统 结构 
现在 ,人 们 实际 使 用 的 操作 系统 多 数 都 采用 层次 式 结构 ,如 UNIX 系统 的 核心 层 就 是 


采用 层次 式 结构 。 层 次 式 结构 既 具 有 上 述 单 体 结构 的 优点 ,又 有 新 的 长 处 : 结构 关系 清晰 ， 
能 够 提高 系统 的 可 靠 性 、 可 移植 性 和 可 维护 性 。 

但 是 ,严格 的 分 层 方法 在 实际 设计 上 有 很 多 困难 ,因此 ,目前 大 致 的 分 层 原则 如 下 。 

(1) 把 与 机 器 硬件 相关 的 程序 模块 放 在 最 底层 ,以 起 到 把 其 他 层 与 硬件 隔离 开 的 
作用 。 

(2) 为 了 便于 操作 系统 从 一 种 操作 方式 平滑 地 过 渡 到 另 一 种 操作 方式 ,在 分 层 时 应 该 
把 反映 系统 外 部 特征 的 软件 放 在 最 外 层 。 

(3) 为 进程 或 线程 的 正常 运行 创造 环境 和 提供 条 件 的 内 核 程序 ,如 CPU 调度 、 进 程 或 
线程 的 控制 和 通信 机 构 等 ,应 该 尽 可 能 放 在 最 底层 ,以 支撑 系统 其 他 功能 部 件 的 执行 。 

(4) 尽量 按照 实现 操作 系统 命令 时 模块 间 的 调用 次 序 或 按 进程 间 单 向 发 送信 息 的 顺序 
来 分 层 。 


1.6.3 虚拟 机 结构 


虚拟 机 系统 的 最 早 应 用 是 IBM 公司 的 CP/CMS, 后 来 更 名 为 YM/370。VM/370 的 核 
心 被 称 为 虚拟 监控 程序 , 它 在 裸 机 上 运行 并 且 具 备 多 道 程序 的 功能 。 该 系统 向 上 提供 了 若 
干 台 虚拟 机 ,如 图 1-10 所 示 。 


若干 个 370 虚 拟 机 


大: | 
lasts 
进程 进程 进程 | CMS CMS CMS 
内 核 内 核 内 核 1/O0 指 令 陷 入 VM/370 
虚拟 机 
硬件 370 裸 机 
(a) 虚拟 机 概念 结 (b) 带 CMS 的 VM/370 


图 1-10 虚拟 机 和 VM/370 结构 图 


不 同 于 其 他 操作 系统 的 是 ,这 些 虚拟 机 不 是 具有 文件 管理 等 优良 特征 扩展 的 计算 机 , 它 
们 仅仅 是 精确 复制 的 裸 机 硬件 ,包括 核心 态 /用 户 态 .I/O 功能 、 终 端 等 其 他 真实 硬件 所 具有 
的 功能 。 


因为 每 台 虚 拟 机 都 与 裸 机 相同 ,所 以 每 台 虚 拟 机 都 可 以 运行 一 台 裸 机 所 能 够 运行 的 任 
何 类 型 的 操作 系统 。 不 同 的 虚拟 机 可 以 运行 不 同 的 操作 系统 ,而 且 实 际 如 此 。 
例如 , 某 些 虚拟 机 运行 OS/360 的 后 续 版 本 作为 批 处 理 或 事务 处 理 ,同时 , 另 一 些 运行 
个 单 用 户 交 互 系 统 供 分 时 用 户 使 用 ,该 系统 被 称 为 会 话 监控 系统 (Conversational 
Monitor System,CMS) 。CMS 的 程序 在 执行 系统 调用 时 ,其 系统 调用 陷入 其 虚拟 机 中 的 操 
作 系 统 , 而 不 是 调用 VM/370, 这 就 像 在 真实 的 计算 机 上 一 样 ,然后 CMS 发 出 正常 的 硬件 
1/O 指令 来 执行 该 系统 调用 。 这 些 1/O 指令 被 VM/370 捕获 ,随后 VM/370 执行 这 些 指 
令 。 作 为 对 真实 硬件 模拟 的 一 部 分 ,通过 将 多 道 程序 功能 和 提供 虚拟 机 分 开 实现 ,它们 更 简 
单 、 更 灵活 且 更 易于 维护 。 


1.6.4 OS 结构 


C/S 结构 , 即 人 们 熟知 的 客户 机 和 服务 器 结构 。 它 是 软件 系统 体系 结构 ,通过 它 可 以 充 
分 利用 两 端 硬件 环境 的 优势 ,将 任务 合理 分 配 到 Client 端 和 Server 端 来 实现 ,降低 了 系统 
的 通信 开销 。 
目前 大 多 数 应 用 软件 系统 都 是 Client/Server 形式 的 两 层 结构 ,由 于 现在 的 软件 应 用 系 
统 正在 向 分 布 式 的 Web 应 用 发 展 ,Web 和 Client/Server 应 用 都 可 以 进行 同样 的 业务 处 理 ， 
应 用 不 同 的 模块 共享 逻辑 组 件 ; 因此 ,内 部 的 和 外 部 的 用 户 都 可 以 访问 新 的 和 现 有 的 应 用 
系统 ,通过 现 有 应 用 系统 中 的 逻辑 可 以 扩展 出 新 的 应 用 系统 。 这 也 就 是 目前 应 用 系统 的 发 
展 方向 。 

C/S 结构 的 基本 原则 是 将 计算 机 应 用 任务 分 解 成 多 个 子 任务 ,由 多 台 计 算 机 分 工 完成 ， 
即 采 用 “功能 分 布 ? 原 则 。 客 户 端 完成 数据 处 理 , 数 据 表 示 以 及 用 户 接口 功能 ;服务 器 端 完 
成 DBMS( 数 据 库 管 理 系统 ) 的 核心 功能 。 这 种 客户 请 求 服务 .服务 器 提供 服务 的 处 理 方式 
是 一 种 新 型 的 计算 机 应 用 模式 。 


1.6.5 微 内 核 架 构 


微 内 核 ( 英 文中 常 译 作 & micro-kernel 或 者 micro kernel) ,是 一 种 能 够 提供 必要 服 
务 的 操作 系统 内 核 ; 其 中 这 些 必 要 的 服务 包括 任务 .线程 .交互 进程 通信 (Inter-Process 
Communication,IPC) 以 及 内 存 管理 等 。 所 有 服务 (包括 设备 驱动 ) 在 用 户 模式 下 运行 ,而 处 
理 这 些 服 务 同 处 理 其 他 的 任何 一 个 程序 一 样 。 因 为 每 个 服务 只 是 在 自己 的 地 址 空间 运行 ， 
所 以 这 些 服 务 彼 此 之 间 都 受到 了 保护 。 

微 内 核 是 内 核 的 一 种 精简 形式 。 将 通常 与 内 核 集成 在 一 起 的 系统 服务 层 分 离 出 来 , 变 
成 可 以 根据 需求 加 入 的 选 件 , 这 样 就 可 提供 更 好 的 可 扩展 性 和 更 加 有 效 的 应 用 环境 。 使 用 
微 内 核 设 计 , 对 系统 进行 升级 ,只 要 用 新 模块 蔡 换 旧 模 块 ,不 需要 改变 整个 操作 系统 。 

可 以 用 商业 对 比 来 解释 微 内 核 的 模块 概念 。 考 虑 一 个 过 度 忙碌 的 商务 经 理 , 通 过 将 工 
作 分 给 其 他 人 ,这 位 经 理 可 以 将 他 的 能 力 更 有 效 地 用 于 重要 的 商务 工作 中 去 ,并 集中 于 其 他 
一 些 任务 ,例如 开辟 新 的 商务 分 支 等 。 可 以 雇用 一 些 新 人 来 支持 增长 的 商务 活动 。 经 理 协 
调 这 些 工 作 , 但 由 其 他 的 人 做 好 雇佣 他 们 时 说 好 要 做 的 事 。 与 此 类 似 , 微 内 核 操 作 系 统 支持 
执行 少量 核心 任务 ,并 管理 可 安装 模块 的 活动 。 用 这 种 方式 , 微 内 核对 于 它 能 做 的 工作 是 非 
常 有 效 的 ,并 是 可 移植 的 ,这 是 指 它 可 以 被 设计 成 在 不 同 的 处 理 器 上 运行 。 


操作 系统 引 论 


姓 一 避 


握 作 系统 原理 


基于 微 内 核 的 操作 系统 具有 如 下 特征 。 

微 内 核 提 供 一 组 “最 基本 ”的 服务 ,如 进程 调度 、 进 程 间 通信 、 存 储 管理 处理 1/O 设备 。 
其 他 服务 ,如 文件 管理 .网 络 支持 等 通过 接口 连 到 微 内 核 。 与 此 相反 ,内 核 是 集成 的 , 比 微 内 
核 更 大 。 

微 内 核 具 有 很 好 的 扩展 性 ,并 可 简化 应 用 程序 开发 。 用 户 只 运行 他 们 需要 的 服务 ,这 有 
利于 减少 磁盘 空间 和 存储 器 需求 。 

微 内 核 的 目标 是 将 系统 服务 的 实现 和 系统 的 基本 操作 规则 分 离开 来 。 例 如 ,进程 的 输 
入 /输出 锁定 服务 可 以 由 运行 在 微 内 核 之 外 的 一 个 服务 组 件 来 提供 。 这 些 非常 模块 化 的 用 
户 态 服务 器 用 于 完成 操作 系统 中 比较 高 级 的 操作 ,这 样 的 设计 使 内 核 中 最 内 核 的 部 分 的 设 
计 更 简单 。 一 个 服务 组 件 的 失效 并 不 会 导致 整个 系统 的 崩溃 ,内 核 需要 做 的 ,仅仅 是 重新 启 
动 这 个 组 件 ,而 不 必 影 响 其 他 的 部 分 。 

IBM 、Microsoft、 开 放 软 件 基 金 会 (OSF) 和 UNIX 系统 实验 室 (USL) 等 新 操作 系统 都 
利用 了 这 一 研究 成 果 的 优点 。 


小 结 


一 个 完整 的 计算 机 系统 主要 是 由 硬件 和 软件 组 成 的 。 硬 件 是 软件 建立 与 活动 的 基础 ， 
而 软件 是 对 硬件 功能 的 扩充 。 

从 传统 意义 上 讲 , 操 作 系统 的 基本 类 型 有 批 处 理 系 统 、 分 时 系统 和 实时 系统 。 各 种 操作 
系统 有 着 不 同 的 设计 目标 ,具有 不 同 的 性 能 。 但 操作 系统 作为 整体 ,有 自己 的 基本 特征 , 即 
并 发 性 ,共享 性 、 虚 拟 性 和 异步 性 。 

操作 系统 是 裸 机 上 的 第 一 层 系 统 软 件 , 它 向 下 管理 系统 中 的 各 种 资源 ,向 上 为 用 户 和 程 
序 提供 服务 。 一 般 来 说 ,操作 系统 有 如 下 几 种 体系 结构 : 单 体 结构 、 层 次 结构 、 虚 拟 结构 、 
C/S 结构 。 


第 2 章 进程 与 线程 


处 理 器 管理 是 操作 系统 的 重要 组 成 部 分 ,负责 管理 .调度 和 分 配 计算 机 系统 的 重要 资 
源 一 一 处 理 器 ,并 控制 程序 的 执行 。 程 序 以 进程 的 形式 来 占用 处 理 器 和 系统 资源 ,处 理 器 管 
理 中 最 重要 的 是 处 理 器 调度 , 即 进程 调度 ,也 就 是 控制 ,协调 进程 对 处 理 器 的 竞争 。 

本 章 详 细 论 述 进 程 和 线程 的 基本 概念 及 其 实现 ,不 同类 型 的 操作 系统 可 能 采取 不 同 的 
调度 策略 ,在 介绍 进程 和 线程 的 基础 上 ,全 面 讨论 各 个 层次 的 处 理 器 调度 方法 。 


2.0 问题 导入 


现代 操作 系统 最 主要 的 特点 在 于 实现 多 道 程 序 并 发 执行 ,并 由 此 引发 资源 共享 。“ 程 

序 ” 是 一 个 在 时 间 上 严格 有 序 的 指令 集合 。 程 序 规定 了 完成 某 一 任务 时 ,计算 机 所 需 做 的 各 

种 操作 ,以 及 这 些 操 作 的 执行 顺序 。 如 果 不 对 人 们 熟悉 的 “程序 ”概念 加 以 扩充 ,就 无 法 刻画 

多 个 程序 共同 运行 时 出 现 的 特征 。 举 一 个 简单 的 例子 : 编译 程序 P 编译 原 程序 甲 ,从 A 点 

开始 工作 ,执行 到 B 点 时 将 信息 记 到 磁盘 上 , 且 程 序 P 在 B 点 等 待 磁盘 传输 ,为 了 提高 系统 效 

率 ,在 多 道 环境 下 ,让 编译 程序 P 再 为 原 程序 乙 进行 编译 , 仍 从 A 点 开始 工作 ,如 图 2-1 所 示 。 
编译 程序 P 


A 一 一 人 (P 的 入 口 ， 处 理 源 
程序 乙 ) 


源 程 
序 甲 序 乙 


《(P 把 源 程序 甲 的 信 
B 一 一 信息 记 盘 等 磁盘 完成 ) 


图 2-1 编译 程序 P 的 状态 


现在 怎样 来 描述 编译 程序 P 的 状态 呢 ? 称 它 在 B 点 等 待 磁盘 传输 状态 ,还 是 称 它 正在 
从 A 点 开始 执行 的 状态 ? 可 见 程序 与 计算 不 再 一 一 对 应 。 如 何 描述 程序 在 并 发 执行 时 对 
系统 资源 的 共享 ,以 及 表现 出 程序 执行 的 动态 过 程 ,在 学 完 本 章 后 ,读者 很 容易 就 能 找到 
答案 。 


青 作 系统 原理 


2.1.1 进程 的 引入 


相对 于 单 道 批 处 理 系统 ,在 多 道 批 处 理 系统 中 ,多 个 作业 在 内 存 中 工作 过 程 的 管理 将 更 
加 困难 。 程 序 在 单 道 运行 方式 和 多 道 运行 方式 下 ,具有 不 同 的 执行 次 序 ,也 导致 了 各 自 不 同 
的 特点 。 为 了 准确 描述 多 个 程序 之 间 的 执行 次 序 , 先 介绍 一 种 图 形 工具 一 一 前 趋 图 。 

前 趋 图 (Precedence Graph) 是 一 个 有 向 无 循环 图 (Directed Acyclic Graph,DAG), 用 于 
描述 程序 、 程 序 段 或 语句 执行 的 先后 次 序 , 图 中 每 个 结 点 可 以 表示 一 条 语句 、 一 个 程序 段 或 
一 个 进程 , 结 点 闻 的 有 向 边 * 一 ”表示 两 个 结 点 之 间 的 前 趋 关 系 : 习 二 (Pi,P;)|P; 必须 在 
P; 开始 执行 前 完全 执行 完 }。 如 果 (Pi,P;)E 一 ,可 写成 Pi 一 已 ,Pi 是 P; 的 直接 前 趋 ,P; 
是 P; 的 直接 后 继 。 把 没有 前 驱 的 结 点 称 为 初始 结 点 ,把 没有 后 继 的 结 点 称 为 终止 结 点 。 

图 2-2 给 出 了 一 个 具有 9 个 结 点 的 前 趋 图 ,该 前 趋 图 中 存在 下 面 的 前 趋 关 系 : 

Pi 一 P:,Pi 一 P:,Pi 一 Pi,P: — Ps,P; — Ps,P,—> Ps, 
是 


或 表示 为 : 
P= {Pi,P;,P;,P,,Ps,Ps,P;,Ps.,P,)} 
一 一 {(Pi,P;,),(Pi,P;),(Pi,P,),(P;,,P:).(P;,Ps),(P,,Pse),(P,,P;), 
(Ps ,Ps),(Ps ,Ps),(P; ,Ps,),(Ps,P,)} 
例如 ,对 于 如 下 具有 4 条 语句 的 程序 段 : 
Pi:a= z++l1; Pe: b= y++2; Pi:c 一 4a 十 0 Pi: d= c+t3; 
假设 执行 单元 是 语句 ,可 画 出 如 图 2-3 所 示 的 前 趋 图 ,来 描述 这 4 条 语句 之 间 的 执行 次 
序 。 其 中 , P; 因为 必须 在 a 和 2 都 被 赋值 后 才 可 执行 ,因此 有 Pi 一 Ps 同时 Ps 一 P3; 同时 ， 
Ps 必须 在 c 被 赋值 后 才 可 执行 ,因此 有 Ps 一 P,; 而 P! 和 了 :因为 彼此 之 间 互 不 依赖 ,所 以 
没有 前 趋 关 系 , 谁 先 执行 或 后 执行 都 可 以 。 


@、 


人 @@ 
Cu 一 -他 (一 他 
@ 四 
Cy 的 一- 节 
加 中 
图 2-2 9 结 点 前 趋 图 示例 图 2-3 4 结 点 前 趋 示例 图 


在 了 解 前 趋 图 的 基础 上 ,下 面 主 要 介绍 程序 单 道 顺序 执行 与 多 道 并 发 执行 的 特点 ,以 引 
入 进程 的 概念 。 

1. 程序 的 顺序 执行 

在 单 道 方 式 下 ,程序 处 在 一 个 顺序 的 环境 中 ,多 个 程序 在 这 一 环境 中 是 顺序 执行 的 。 如 


图 2-4 所 示 , 两 个 作业 之 间 的 运行 方式 就 是 单 道 顺 序 执行 的 方式 。 
作业 1 作业 2 


图 2-4 作业 的 单 道 顺 序 执行 


程序 的 顺序 执行 具有 如 下 一 些 特征 。 

(1) 顺序 性 。 处 理 器 严格 按照 程序 所 规定 的 顺序 执行 , 即 每 个 操作 必须 在 前 一 个 操作 
结束 之 后 开始 。 

(2) 封闭 性 。 程 序 在 封闭 的 环境 下 运行 , 即 程序 运行 时 独占 全 部 系统 资源 。 

(3) 可 再 现 性 。 只 要 程序 的 初始 条 件 相同 , 则 其 执行 结果 相同 ,无 论 何 时 执行 ,也 无 论 
运行 速度 怎样 ,例如 键盘 输入 的 速度 是 快 还 是 慢 , 都 不 会 影响 最 后 的 运行 结果 。 

程序 单 道 执行 之 所 以 具有 以 上 特点 ,是 因为 单 道 环境 下 作业 独占 了 系统 资源 。 

2. 程序 的 并 发 执行 

在 多 道 方式 下 ,程序 处 在 一 个 并 发 的 执行 环境 中 ,多 个 程序 在 这 一 环境 中 是 并 发 执行 


的 , 即 若干 个 程序 段 同时 在 系统 中 运行 ,这 些 程序 的 执行 在 时 间 上 是 重 倒 的 ,一 个 程序 段 的 
执行 尚未 结束 , 另 一 个 程序 段 的 执行 已 经 开始 。 如 图 2-5 所 示 , 两 个 程序 投入 系统 中 的 并 发 
执行 方式 ,阴影 部 分 所 表示 的 时 间 就 是 它们 并 发 执行 的 时 间 。 


抽 


图 2-5 程序 的 并 发 执行 


为 了 实现 并 发 ,必须 共享 系统 中 的 资源 ,如 CPU 轮流 分 配给 各 个 程序 .打印 机 依次 打 
印 排队 作业 的 信息 、 每 个 作业 占用 磁盘 和 内 存 的 一 部 分 空间 等 。 这 样 程序 的 并 发 执行 和 系 
统 资源 的 共享 就 使 得 操作 系统 的 工作 变 得 很 复杂 ,不 像 单 道 程序 执行 时 那样 简单 直观 ,所 
具有 的 特征 也 是 和 程序 的 顺序 执行 所 不 同 的 。 程 序 的 并 发 执行 具有 如 下 一 些 特征 。 

(1) 间断 性 。 并 发 执行 中 的 程序 可 能 由 于 对 资源 的 竞争 和 共享 ,或 者 相互 协作 共同 完 
成 某 个 任务 而 存在 着 相互 制约 的 关系 ,也 可 能 在 运行 时 由 于 一 个 时 间 片 用 完 而 中 断 ,放弃 
CPU 使 自己 无 法 继续 运行 下 去 。 但 是 当 别 的 程序 释放 资源 ,或 者 再 次 被 CPU 选中 时 , 它 又 
能 继续 运行 下 去 。 这 使 得 并 发 执行 的 程序 具有 “执行 -暂停 -执行 ”的 活动 规律 。 一 个 程序 的 
执行 由 于 是 间断 性 的 ,一 次 一 个 程序 可 能 需要 多 次 执行 才能 完成 , 即 一 个 程序 可 以 对 应 多 次 
执行 ,程序 和 执行 不 再 一 一 对 应 。 

(2) 失去 封闭 性 。 并 发 执行 的 多 个 程序 ,由 于 资源 共享 以 及 相互 协作 ,打破 了 程序 单 道 
执行 时 所 具有 的 封闭 性 。 由 于 失去 了 封闭 性 ,资源 的 使 用 状态 不 再 由 某 个 程序 所 决定 ,而 是 
受到 并 发 程序 的 共同 影响 。 

(3) 不 可 再 现 性 。 由 于 失去 了 封闭 性 ,多 个 程序 并 发 执行 时 的 相对 速度 是 不 确定 的 ,每 
个 程序 都 会 经 历 “ 停 停 走 走 ” 的 过 程 。 但 何 时 发 生 控制 转换 并 非 完全 由 程序 本 身 决 定 ,而 是 


进程 与 线程 


媳 避 


楷 作 系统 原理 


与 整个 系统 当时 所 处 的 环境 有 关 , 具 有 一 定 的 随机 性 ,因而 也 失去 了 可 再 现 性 ,可 能 发 生 “ 与 
时 间 有 关 的 错误 ”。 

举例 来 说 ,为 了 了 解 某 单行 道 的 交通 流量 ,在 路 口 安放 一 个 监视 器 ,功能 是 有 车 经 过 
该 路 段 时 ,就 向 计算 机 发 送 一 个 信号 。 为 计算 机 系统 设计 两 个 程序 : 程序 A 的 功能 是 接 
收 到 监视 器 的 信号 时 ,就 在 计算 单元 COUNT 上 加 1; 程序 B 的 功能 是 每 阳 半 个 小 时 ,将 
计数 单元 COUNT 的 值 打印 出 来 ,然后 清 零 。COUNT 初始 时 为 0。 两 个 程序 的 描述 
如 下 。 


程序 A: 程序 B: 
while(1) while(1) 
{ 
RM: 收 到 监视 器 的 信号 ; B1: 延迟 半 小 时 ; 
A2: COUNT = COUNT + 1; B2: 打印 COUNT 的 值 ; 
B3: COUNT= 0; 


} 


因为 现在 是 多 道 程序 设计 环境 ,程序 A 和 程序 B 的 执行 可 能 是 间断 性 的 ,这 些 程序 
的 执行 被 交织 在 一 起 ,没有 任何 规律 可 循 。 在 它们 之 间 不 排除 会 有 这 样 的 执行 顺序 发 
生 : Al1 一 A2 一 Bl 一 B2 一 Al 一 A2 一 B3。 假 定 系 统 在 发 生 这 一 顺序 前 ,情况 一 直 正 常 ， 
COUNT 的 值 是 9。 按 照 刚才 的 执行 顺序 .Al 收 到 监视 器 发 来 的 第 10 辆 车 通过 的 消息 ,于 
是 A2 在 COUNT 上 完成 加 1 的 操作 ,计数 器 COUNT 取 值 为 10。 紧 接着 在 Bl 延迟 半 小 
时 后 ,由 B2 将 COUNT 中 的 值 打印 出 来 。 这 时 又 执行 Al, 它 收 到 监视 器 发 来 的 第 11 辆 车 
通过 的 消息 ,通过 A2 计数 器 COUNT 取 值 为 11。 这 时 执行 B3, 把 COUNT 清 零 ,结果 第 
11 辆 车 并 没有 被 打印 出 来 , 少 计算 了 一 台 车 ,但 如 果 不 是 这 样 的 执行 顺序 ,这 种 现象 可 能 就 
不 会 出 现 ,由 于 失去 了 封闭 性 ,并 发 程序 之 间 会 互相 影响 ,无 法 清楚 何 时 会 出 现 什么 样 的 执 
行 顺序 ,结果 的 可 再 现 性 也 就 不 复 存 在 了 。 

可 见 , 在 多 道 程 序 设计 环境 下 ,“ 程 序 ”? 具 有 了 与 单 道 程序 设计 环境 下 截然 不 同 的 特性 。 
程序 的 执行 出 现 了 “ 停 停 走 走 ” 的 新 状态 。 而 程序 本 身 是 机 器 能 翻译 或 执行 的 一 组 动作 或 指 
令 , 或 者 写 在 纸 上 ,或 者 存放 在 磁盘 等 介质 上 ,是 静止 的 。 直 接 从 程序 的 字面 上 无 法 看 出 它 
什么 时 候 运 行 ,什么 时 候 停顿 ,也 看 不 出 它 是 否 被 其 他 程序 影响 或 者 影响 其 他 程序 。 很 显 
然 ,程序 这 个 静态 的 概念 无 法 刻画 程序 并 发 执行 过 程 中 的 特征 。 因 此 ,在 操作 系统 里 ,以 “ 程 
序 ” 为 基础 ,引入 了 进程 (Process) 这 一 新 的 概念 。 


2.1.2 进程 与 进程 控制 块 


1. 进程 的 概念 

“进程 "是 操作 系统 中 最 基本 最 重要 的 概念 ,是 在 多 道 程序 系统 出 现 后 ,为 了 刻画 系统 内 
部 的 运动 状况 ,描述 运行 程序 的 活动 规律 而 引进 的 新 概念 。 进 程 的 概念 最 早 是 1960 年 在 美 
麻 省 理工 学 院 MULTICS 和 IBM 公司 CTSS/360 提出 和 实现 的 。 

直到 目前 为 止 ,进程 还 没有 非常 确切 和 统一 的 描述 。 但 操作 系统 引入 进程 的 目的 是 明 

确 的 。 一 是 刻画 系统 的 动态 性 ,发 挥 系 统 的 并 发 性 。 二 是 解决 共享 性 ,正确 地 描述 程序 的 执 
行 状态 。 综 上 给 出 进程 的 定义 : 进程 是 可 并 发 执行 的 程序 在 某 个 数据 集合 上 的 一 次 计算 活 
动 ,也 是 操作 系统 进行 资源 分 配 和 运行 调度 的 基本 单位 。 


可 以 看 出 ,进程 和 程序 有 着 密切 的 关系 ,但 又 是 两 个 完全 不 同 的 概念 ,进程 和 程序 的 关 
系 主要 体现 在 以 下 几 点 。 

(1) 进程 是 一 个 动态 概念 ,而 程序 是 一 个 静态 概念 。 进 程 存在 于 程序 的 执行 过 程 中 。 
程序 一 旦 执行 ,进程 就 会 被 创建 ,该 进程 因 调 度 而 执行 , 因 得 不 到 资源 而 暂时 停止 执行 , 因 执 
行 结束 而 撤销 ,因而 具有 生命 周期 ,会 动态 地 产生 和 消亡 。 程 序 是 指令 的 有 序 集合 ,不 能 深 
刻 揭示 并 发 程序 间 的 内 在 活动 联系 及 状态 的 变化 。 它 作为 静态 的 实体 而 存在 ,可 以 作为 程 
序 文件 被 长 久保 存 。 

(2) 进程 具有 并 发 特性 ,而 程序 没有 。 进 程 不 仅 包 括 程序 .数据 ,还 有 一 系列 描述 其 活 
动情 况 的 数据 结构 。 进 程 能 够 通过 数据 结构 由 操作 系统 动态 地 描述 其 并 发 执行 过 程 中 的 状 
态 和 信息 ,是 一 个 独立 运行 的 单位 ,能 与 其 他 进程 并 发 执行 ,具有 并 发 执行 的 特性 ,能 够 确切 
地 描述 并 发 活动 。 程 序 没 有 相应 的 数据 结构 进行 描述 ,不 能 作为 调度 执行 单位 , 仅 代表 一 组 
语句 的 集合 ,所 以 不 具备 这 种 特性 。 

(3) 进程 间 会 相互 制约 ,而 程序 没有 。 多 道 环 境 下 ,进程 在 对 资源 共享 和 竞争 中 必然 会 
相互 制约 ,造成 各 自前 进 速度 的 不 可 预见 性 。 程 序 本 身 是 静态 的 ,不 存在 这 种 特性 。 

(4) 进程 与 程序 之 间 存 在 多 对 多 的 联系 。 一 个 程序 的 多 次 运行 分 别 对 应 不 同 的 进程 。 
程序 每 运行 一 次 ,操作 系统 为 其 创建 一 个 进程 , 若 该 程序 运行 多 次 , 则 每 一 次 对 应 一 个 不 同 
的 进程 ,所 以 程序 对 进程 存在 1 : n 的 联系 。 一 个 进程 可 以 通过 执行 某 个 指定 程序 来 与 不 同 
的 程序 关联 , 即 一 个 进程 在 其 活动 中 可 以 顺序 地 执行 若干 程序 ,所 以 进程 对 程序 存在 1 :nn 
的 联系 。 

2. 进程 的 特征 

通过 以 上 分 析 ,进程 具有 如 下 基本 特征 。 

(1) 动态 性 。 进 程 是 程序 在 数据 集合 上 的 一 次 执行 过 程 ,是 动态 的 概念 ,具有 生命 周 
期 ,由 创建 而 产生 ,由 调度 而 执行 ,由 事件 而 等 待 ,由 撤销 而 消亡 。 

(2) 并 发 性 。 进 程 的 执行 可 以 在 时 间 上 有 所 重 且 ,进程 的 执行 是 可 以 被 打 断 的 , 即 进程 
在 执行 完 一 条 指令 且 下 一 条 指令 未 执行 前 ,可 能 需要 被 迫 让 出 CPU, 由 其 他 进程 执行 若干 
条 指令 后 才能 重新 获得 CPU 继续 执行 。 

(3) 独立 性 。 进 程 是 系统 中 资源 分 配 、 保 护 和 调度 的 基本 单位 ,说 明 它 具有 独立 性 , 凡 
是 未 建立 进程 的 程序 ,都 不 能 作为 独立 单位 参与 调度 和 运行 。 此 外 ,每 个 进程 都 可 以 有 各 自 
独立 的 ,不 可 预知 的 速度 在 CPU 上 推进 ,这 也 表现 出 进程 的 独立 性 。 

(4) 异步 性 。 各 进程 向 前 推进 的 速度 是 不 可 预知 的 , 即 以 异步 方式 运行 ,这 造成 进程 间 
的 相互 制约 ,使 程序 执行 失去 再 现 性 。 

(5) 结构 性 。 进 程 有 一 定 的 结构 , 它 由 程序 ,数据 和 控制 结构 (如 进程 控制 块 ) 等 组 成 。 
程序 规定 了 该 进程 所 要 执行 的 任务 ,数据 是 程序 操作 的 对 象 ,而 控制 结构 中 含有 进程 的 描述 
信息 和 控制 信息 ,是 进程 中 最 关键 的 部 分 。 

3. 进程 控制 块 的 概念 及 其 内 容 

从 上 面 的 特性 可 以 看 出 ,一 个 进程 创建 后 ,需要 有 自己 对 应 的 程序 和 该 程序 运行 时 所 需 
的 数据 ,但 仅 有 程序 和 数据 还 不 行 ,还 需要 数据 结构 来 刻画 进程 的 动态 特征 ,描述 进程 状态 、 
占有 资源 情况 .调度 信息 等 ,通常 使 用 一 种 称 为 进程 控制 块 (Process Control Block,PCB) 的 
数据 结构 来 刻画 。 这 样 ,一 个 进程 要 由 三 个 部 分 组 成 : 程序 .数据 集合 以 及 进程 控制 块 。 由 


进程 与 线程 


媳 凡 直 


握 作 系统 原理 


于 进程 的 状态 在 不 断 发 生变 化 , 某 时 刻 进程 的 内 容 及 其 状态 集合 称 为 进程 映像 (Process 
Image) ,其 包括 进程 控制 块 ,进程 程序 段 ,进程 核心 栈 和 进程 数据 段 4 个 要 素 。 

进程 控制 块 是 随 着 进程 的 创建 而 建立 , 随 着 进程 的 撤销 而 取消 的 ,因此 系统 是 通过 
PCB 来 “感知 ”一 个 个 进程 的 ,PCB 是 进程 存在 的 唯一 标志 。 每 个 进程 有 且 仅 有 一 个 进程 控 
制 块 ,是 操作 系统 用 来 记录 和 刻画 进程 状态 及 有 关 信 息 的 数据 结构 ,是 进程 动态 特征 的 一 种 
汇集 ,也 是 操作 系统 掌握 进程 的 唯一 资料 结构 和 管理 进程 的 主要 依据 。 进 程控 制 块 应 常 驻 
内 存 , 其 包括 进程 执行 时 的 情况 以 及 进程 让 出 CPU 之 后 所 处 的 状态 、 断 点 等 信息 ,一 般 来 
说 应 该 包含 以 下 信息 ,如 图 2-6 所 示 。 


进程 标识 1D 进程 状态 
_ 进程 组 标识 人 D 进程 优先 级 
识 信息 信息 
人 用 户 进程 名 人 等 待 原因 
用 户 组 名 进程 已 占用 和 等 待 CPU 的 时 间 
通用 寄存 器 程序 段 和 数据 段 指针 
, 指令 计数 器 进程 之 间 的 族 系 信息 
信息 凡 息 
油 本 程序 状态 字 Wii 进程 的 同步 和 通信 所 需 的 机 制 
用 户 栈 指针 进程 已 分 配 的 资源 清单 和 链接 指针 


图 2-6 进程 控制 块 PCB 的 组 成 


(1) 标识 信息 。 进 程 标识 符 用 于 唯一 地 标识 一 个 进程 。 分 为 用 户 使 用 的 外 部 标识 符 和 
系统 使 用 的 内 部 标识 符 。 系 统 中 的 所 有 进程 都 被 赋予 唯一 的 、 内 部 使 用 的 数值 型 进程 号 ( 通 
常 是 0 一 32 768 的 正 整数 ) ,操作 系统 内 核 函 数 可 通过 进程 号 来 引用 PCB。 常 用 的 标识 信息 
包括 进程 标识 ID .进程 组 标识 ID、 用 户 进 程 名 、 用 户 组 名 等 。 

(2) 现场 信息 。 现 场 信息 主要 是 由 处 理 器 的 各 种 寄存 器 中 的 内 容 组 成 ,保留 进程 在 运 
行 时 存放 在 处 理 器 现场 中 的 各 种 信息 。 当 中 断 时 处 理 器 的 当前 信息 必须 保存 ,确保 进程 重 
新 调用 时 系统 能 从 断 点 继续 执行 。 通 常 , 被 保存 的 信息 包括 通用 寄存 器 、 指 令 计 数 器 、 程 序 
状态 字 、 用 户 栈 指针 等 内 容 。 

(3) 调度 信息 。 进 程 调度 和 进程 对 换 的 相关 信息 。 包 括 进 程 当前 状态 、 进 程 的 优先 级 、 
等 待 原因 、 进 程 已 占用 和 等 待 CPU 的 时 间 等 。 

(4) 控制 信息 。 控 制 信息 用 来 管理 进程 ,包括 程序 段 指针 ,数据 段 指针 ,进程 之 间 的 族 
系 信息 ,进程 的 同步 和 通信 所 需 的 机 制 ( 信 号 量 、 消 息 队 列 指针 等 ), 进 程 已 分 配 的 资源 清单 
和 链接 指针 等 。 

4. 进程 控制 块 的 组 织 方式 

进程 的 特征 主要 由 PCB 来 刻画 ,为 了 便于 对 进程 进行 管理 和 调度 ,常常 将 进程 的 PCB 
通过 某 种 方法 组 织 起 来 ,一 般 来 说 ,有 三 种 通用 的 组 织 方式 : 线性 方式 .链接 方式 和 索引 
方式 。 

(1) 线性 方式 。 操 作 系 统 根据 系统 内 进程 的 最 大 数目 ,静态 地 分 配 主 存 中 的 某 块 空间 ， 
所 有 进程 的 PCB 都 组 织 在 一 个 线性 表 中 。 线 性 方式 的 优点 是 简单 易 行 ,缺点 是 限定 系统 中 
的 进程 最 大 数 ,经 常 要 扫描 整个 线性 表 , 调 度 效 率 较 低 。 

(2) 链接 方式 。 利 用 链接 指针 将 同一 状态 的 进程 控制 块 链接 为 一 个 队列 ,如 图 2-7 所 
示 。 根 据 进程 的 状态 可 以 形成 不 同 状态 的 队列 。 


PCB1 PCB2 PCB 
| _Pce2 | 加 


图 2-7 链接 方式 


(3) 索引 方式 。 索 引 方式 是 线性 方式 的 一 种 改进 ,利用 索引 表 记 录 不 同 状 态 进 程 的 
PCB 地 址 ,系统 建立 若干 个 索引 表 ,状态 相同 进程 的 PCB 组 织 在 同一 张 索引 表 中 ,每 个 索引 
表 的 表 目 中 存放 PCB 在 PCB 表 中 的 编号 或 地 址 ,各 个 索引 表 在 主 存 中 的 起 始 地 址 放 在 内 
核 的 专用 指针 单元 ,如 图 2-8 所 示 。 


就 绪 索 引 表 上 | 


就 绪 表 指针 


图 2-8 索引 方式 


进程 控制 块 是 操作 系统 中 最 重要 的 数据 结构 , 它 包含 操作 系统 所 需要 的 关于 进程 的 所 
有 信息 。 


2.2 进程 控制 


操作 系统 的 一 个 主要 职责 就 是 控制 进程 的 执行 。 通 过 进程 的 特性 ,进程 要 么 正在 执行 ， 
要 么 没有 执行 ,这 样 一 个 进程 就 有 两 种 基本 状态 : 运行 (Running) 和 非 运行 (Not-running)。 
如 何 控制 这 些 进程 ,如 何 完善 简单 的 两 种 状态 ,在 了 解 层 次 结构 的 基础 上 ,有 必要 讨论 进程 
的 创建 和 终止 ,在 进程 的 整个 生命 周期 中 介绍 其 状态 及 状态 转换 的 实现 。 

对 进程 的 控制 和 管理 由 系统 中 的 原 语 来 实现 。 原 语 (Primitive) 是 完成 系统 特定 功能 的 
不 可 分 割 的 过 程 , 它 具 有 原子 性 操作 ,其 程序 段 不 可 以 被 中 断 ,或 者 说 原 语 不 能 并 发 执行 。 


2.2.1 进程 的 层次 结构 


不 同 操作 系统 创建 进程 的 方式 不 尽 相 同 ,传统 UNIX 系统 中 是 通过 一 个 进程 创建 另 一 
个 进程 ,通常 把 创建 进程 的 进程 称 为 父 进程 ,而 把 被 创建 的 进程 称 为 子 进程 , 子 进程 继承 和 
共享 父 进 程 的 资源 。 子 进程 可 以 继续 创建 更 多 的 子孙 进程 ,由 此 便 形成 了 进程 的 层次 结构 ， 
组 成 进程 家 族 。 在 Windows 中 ,新 老 进 程 之 间 不 存在 族 系 关系 或 层次 关系 ,两 者 是 平等 的 。 

当 进 程 之 间 存 在 层次 结构 时 ,了 解 进程 间 的 这 种 关系 是 十 分 重要 的 。 因 为 子 进程 的 行 
为 完全 由 父 进程 和 默认 行为 所 决定 ,把 自己 的 地 址 空间 制作 一 个 副本 ,其 中 包括 正文 段 . 数 
据 段 .用 户 栈 和 核心 栈 , 即 子 进程 继承 父 进 程 的 全 部 资源 ,如 果子 进程 需要 ,可 以 使 用 系统 调 
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让 新 程序 替换 老 程 序 , 此 后 ,父子 进程 可 以 自行 其 道 。 当 子 进程 被 撤销 时 ,应 将 其 从 父 进 
程 获得 的 资源 归还 给 父 进程 , 当 父 进 程 被 撤销 时 ,也 必须 同时 撤销 所 有 的 子 进程 。 

为 了 形象 地 描述 进程 的 家 族 关 系 而 引进 了 进程 家 族 树 ,也 称 进程 图 (Process Graph) 。 
进程 图 是 一 种 用 于 描述 进程 家 族 关 系 的 有 向 树 ,如 
图 2-9 所 示 。 

在 图 2-9 中 , 结 点 代表 进程 ,圆圈 中 的 符号 是 进 
程 名 称 。 进 程 P; 创建 了 进程 P; 后 (ij 一 1,2,…， 
13) , 称 P; 是 P; 的 父 进程 。 进 程 Pi 与 P; 的 父子 关 
系 在 图 中 用 一 条 由 进程 P; 结 点 指向 进程 P; 结 点 的 
有 向 边 来 描述 。 由 于 内 存 中 的 进程 之 间 一 般 都 存在 
一 定 的 父子 关系 ,因此 ,作为 一 种 描述 进程 家 族 关 系 
图 2-9 进程 家 族 树 的 有 向 树 ,进程 家 族 树 在 进程 控制 时 是 必要 的 。 


2.2.2 进程 创建 


当 有 一 个 新 进程 要 加 入 当前 进程 队列 时 ,操作 系统 就 产生 一 个 控制 进程 的 数据 结构 并 
为 该 进程 分 配 地 址 空间 。 这 样 ,新 进程 就 产生 了 。 

1. 进程 创建 的 原因 

(1) 用 户 登 录 。 在 一 个 交互 式 环境 中 , 当 一 个 新 用 户 在 终端 输入 登录 命令 后 ,若是 合法 
用 户 ,系统 将 为 该 用 户 建立 一 个 进程 。 

(2) 提供 服务 。 当 运行 中 的 用 户 程序 提出 某 种 请 求 后 ,系统 将 专门 创建 一 个 进程 来 提 
供用 户 所 需要 的 服务 。 例 如 ,用 户 程序 要 求 进行 文件 打印 ,操作 系统 将 为 它 创 建 一 个 打印 进 
程 ,这 样 ,不 仅 可 使 打印 进程 与 该 用 户 进程 并 发 执行 ,而 且 还 便于 计算 为 完成 打印 任务 所 花 
费 的 时 间 。 

(3) 作业 调度 。 在 批 处 理 系 统 中 , 当 作业 调度 程序 按 一 定 的 算法 调度 到 某 作业 时 ,操作 
系统 认为 有 资源 可 运行 , 便 将 该 作业 装 入 内 存 , 为 它 分 配 必要 的 资源 ,并 立即 为 它 创 建 进程 。 

在 以 上 三 种 情况 下 ,都 是 由 系统 内 核 为 之 创建 一 个 新 进程 。 

(4) 应 用 请 求 。 基 于 应 用 进程 的 需求 ,由 它 自己 创建 一 个 新 进程 ,以 便 使 新 进程 以 并 发 
运行 的 方式 完成 特定 的 任务 。 例 如 ,一 个 应 用 程序 需要 不 断 地 从 终端 输入 数据 ,根据 数据 进 
行 计算 以 及 输出 计算 结果 , 则 相应 的 应 用 进程 除 完成 计算 任务 外 ,还 可 以 由 它 创 建 一 个 输入 
进程 和 输出 进程 。 这 样 ,输入 进程 计算 进程 和 输出 进程 三 者 都 可 以 并 发 执行 ,从 而 提高 总 
体 效率 。 

2. 进程 创建 的 过 程 

一 般 来 说 ,进程 的 创建 过 程 如 下 。 

(1) 在 进程 列表 中 增加 一 项 ,申请 一 个 空白 PCB, 为 新 进程 分 配 唯 一 的 进程 标识 符 。 

(2) 为 新 进程 分 配 地 址 空间 ,以 便 容 纳 进 程 实 体 。 由 进程 管理 程序 确定 加 载 至 进程 地 
址 空间 中 的 程序 。 

(3) 为 新 进程 分 配 除 主 存 空间 以 外 的 其 他 各 种 资源 。 

(4) 初始 化 PCB, 如 进程 标识 符 、 处 理 器 初始 状态 、 进 程 优先 级 等 。 

(5) 将 新 进程 插入 到 相对 应 的 队列 。 


(6) 通知 操作 系统 的 某 些 模块 ,如 性 能 检测 程序 。 
2.2.3 进程 终止 


在 任何 计算 机 系统 中 ,进程 必须 有 一 种 方法 以 表明 其 运行 结束 。 一 个 批 处 理 任 务 可 以 
包含 一 条 Halt 指令 或 执行 OS 提供 的 终止 调用 。 

1. 进程 终止 的 原因 

进程 终止 可 分 为 正常 终止 和 非 正常 终止 。 导 致 进程 终止 的 事件 大 致 有 14 种 ,如 表 2-1 
所 示 。 


表 2-1 进程 终止 的 原因 


原 因 说 明 

正常 结束 进程 执行 一 个 操作 系统 调用 以 表示 其 运行 完毕 

运行 超时 进程 运行 时 超过 了 指定 的 最 长 时 间 

等 待 超时 进程 等 待 时 间 超 过 了 某 事件 发 生 的 指定 时 间 

内 存 不 足 进程 需要 的 内 存 大 于 系统 可 提供 量 

越界 进程 试图 对 不 允许 接近 的 区 域 进行 操作 

保护 错误 进程 试图 使 用 不 允许 使 用 的 资源 或 文件 ,或 者 使 用 方式 不 对 。 例 如 ， 
试图 对 一 个 只 读 文件 进行 写 操作 

算数 错误 进程 试图 进行 不 允许 的 计算 ,例如 被 0 除 

1/O 故障 输入 /输出 过 程 中 产生 错误 

非法 指令 进程 试图 执行 一 个 不 存在 的 指令 (通常 是 程序 错误 地 转移 到 数据 区 
域 ,把 数据 当成 了 指令 ) 

特权 指令 用 户 进程 试图 执行 一 条 保留 给 OS 使 用 的 指令 

错误 使 用 数据 数据 类 型 出 错 或 者 数据 未 初始 化 

操作 员 或 OS 干预 由 于 某 些 原 因 ,操作 员 或 OS 终止 了 进程 ,例如 发 生死 锁 

父 进程 终止 父 进程 终止 时 ,操作 系统 会 自动 终止 它 的 所 有 子孙 进程 

父 进 程 请 求 父 进程 拥有 终止 所 有 子孙 进程 的 权限 

2. 进程 终止 的 过 程 


一 旦 上 述 事件 发 生 , 系统 或 进程 将 调用 撤销 原 语 来 终止 进程 或 子 进程 。 具体 的 过 程 
如 下 。 

(1) 根据 终止 进程 的 标识 号 ,从 相应 的 队列 中 查找 并 移 除 它 。 

(2) 车 被 终止 进程 正 处 于 运行 状态 , 则 立即 停止 该 进程 的 运行 ,并 重新 调度 。 

(3) 将 子 进程 所 拥有 的 资源 归还 给 父 进程 或 操作 系统 。 

(4) 若 此 进程 拥有 子 进程 , 先 终止 其 所 有 子 进程 ,以 防止 它们 脱离 控制 。 

(5) 回收 PCB, 并 将 其 归还 至 PCB 池 。 


2.2.4 进程 的 状态 与 转换 


进程 从 因 创 建 而 产生 到 终止 而 消亡 的 整个 生命 周期 中 ,有 时 处 于 运行 状态 ,占用 处 理 器 
执行 。 有 时 处 于 非 运行 状态 ,在 非 运行 状 态 时 有 时 是 因为 分 不 到 处 理 器 而 不 能 运行 ,有 时 虽 
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然 处 理 器 空闲 但 因 等 待 某 个 事件 发 生 而 无 法 执行 .这 一 切 说 明 进程 是 活动 的 且 有 状态 变化 ， 
状态 及 状态 之 间 的 转换 体现 进程 的 动态 性 。 

不 同系 统 设 置 的 进程 状态 数目 不 同 ,我 们 分 别 介绍 典型 的 三 状态 模型 .五 状态 模型 和 七 
状态 模型 ,以 及 它们 对 应 的 状态 转换 过 程 。 

1. 三 状态 模型 

为 了 便于 系统 管理 ,一 般 来 说 ,按照 进程 在 执行 过 程 中 的 不 同情 况 至 少 要 定义 三 种 进程 
状态 ,如 图 2-10 所 示 。 


(1) 执行 态 (Running)。 进 程 占用 处 理 器 并 执行 
的 状态 。 在 单 处 理 器 系统 中 ,由 于 处 理 器 的 唯一 性 ， 
完 至 多 只 能 有 一 个 进程 处 于 执行 状态 。 
(2) 就 绪 态 (Ready)。 进 程 具备 执行 条 件 , 即 已 分 
配 到 除 处 理 器 以 外 的 所 有 必要 的 资源 ,等 待 系统 分 配 
处 理 器 以 便 其 运行 的 状态 。 在 一 个 系统 中 ,可 能 有 多 
个 进程 都 处 于 就 绪 状 态 ,通常 把 这 些 就 绪 进 程 组 织 为 
图 2-10 进程 的 三 种 基本 状态 与 转换 “一 个 或 多 个 队列 , 称 这 些 队 列 为 就 绪 队 列 。 
(3) 阻塞 态 (Blocked)。 也 称 等 待 态 (Wait) 或 睡 
眠 态 (Sleep)。 进 程 不 具备 执行 条 件 , 正 在 等 待 某 个 事件 完成 的 状态 。 在 一 个 系统 中 ,可 能 
同时 有 多 个 进程 处 于 阻塞 态 ,通常 把 这 些 进程 组 织 成 一 个 或 多 个 队列 , 称 这 些 队列 为 阻塞 队 
列 。 在 有 的 系统 中 ,按照 产生 阻塞 的 原因 来 组 织 这 些 进程 : 一 个 原因 对 应 建立 一 个 队列 , 因 
为 同样 原因 阻塞 的 进程 被 放 在 同一 队列 中 。 
处 于 执行 态 的 进程 会 因 出 现 等 待 事件 而 进入 阻塞 态 , 当 等 待 事件 完成 之 后 ,阻塞 态 进 程 
转 入 就 绪 态 ,处 理 器 调度 将 会 引发 执行 态 和 就 绪 态 进程 之 间 的 切换 。 引 起 进程 状态 转换 的 
具体 原因 有 以 下 几 点 。 
(1) 就 绪 态 一 执行 态 。 进 程 调度 程序 根据 调度 算法 把 处 理 器 分 配给 某 个 就 绪 进 程 ,并 
把 控制 转 到 该 进程 ,使 它 由 就 绪 态 变 为 执行 态 , 进 程 投入 运行 。 
(2) 执行 态 一 就 绪 态 。 执 行 中 的 进程 因 所 分 配给 它 的 时 间 片 用 完 而 被 暂停 运行 时 ,该 
进程 便 由 执行 态 回 到 就 绪 态 。 
(3) 执行 态 一 阻塞 态 。 因 发 生 某 事件 而 使 进程 的 执行 受阻 (例如 等 待 文件 的 输入 ) ,使 
得 进程 无 法 继续 执行 ,该 进程 将 由 执行 态 变 为 阻塞 态 。 
(4) 阻塞 态 一 就 绪 态 。 阻 塞 进程 在 所 等 待 的 事件 完成 后 ,并 不 能 立即 投入 运行 ,需要 通 
过 进程 调度 程序 统一 调度 才能 获得 处 理 器 ,此 时 将 该 进程 的 状态 变 为 就 绪 态 继续 等 待 处 
理 机 。 
2. 五 状态 模型 
在 很 多 操作 系统 中 ,增加 两 个 进程 状态 : 新 
建 态 (New) 和 终止 态 (Exit) ,如 图 2-11 所 示 。 
(1) 新 建 态 (New)。 进 程 刚 建立 ,还 没有 被 
OS 提交 到 可 运行 进程 队列 。 
(2) 终止 态 (Exit) 。 进 程 已 正常 或 异常 结束 ， 
被 OS 从 可 运行 进程 队列 中 释放 出 来 。 图 2-11 进程 的 5 种 状态 与 转换 


新 建 态 和 终止 态 对 进程 管理 非常 有 用 。 新 建 态 对 应 于 进程 被 创建 时 的 状态 ,进程 尚 
未 进入 就 绪 队 列 。 有 时 会 根据 系统 性 能 的 要 求 或 主 存 容 量 的 限制 推迟 新 建 态 进程 的 提 
交 。 进 程 从 系统 中 退出 时 会 先 终止 ,然后 进入 终止 状态 。 处 于 终止 态 的 进程 不 再 被 调度 
执行 ,在 与 该 任务 相关 的 表格 或 其 他 信息 抽取 完毕 以 后 ,OS 也 不 必 再 保存 与 该 进程 有 关 
的 信息 。 

在 5 状态 模型 中 ,一 个 进程 在 运行 期 间 , 将 仍然 不 断 地 从 一 种 状态 转换 到 另 一 种 状态 。 
一 个 进程 可 能 多 次 处 于 就 绪 、 执 行 和 阻塞 态 ,但 处 于 新 建 态 和 终止 态 的 次 数 有 且 仅 有 一 次 。 
总 之 ,进程 * 生 ”与 “ 死 ” 都 经 过 一 次 ,中 间 的 状态 可 以 多 次 反复 ,很 好 地 体现 了 进程 的 生命 
周期 。 

下 面 简 单 介 绍 一 下 新 增 的 几 种 状态 转换 。 

(1) 新 建 态 一 就 绪 态 。 对 一 个 处 于 新 建 态 的 进程 ,在 就 绪 队 列 能 够 接纳 新 进程 时 ,将 被 
系统 接收 并 进入 就 绪 队 列 , 此 时 的 进程 状态 就 从 新 建 态 转 为 就 绪 态 。 

(2) 执行 态 一 终止 态 。 对 于 一 个 处 于 执行 状态 的 进程 , 当 其 正常 执行 结束 ,或 者 因为 发 
生 了 某 种 事件 而 被 异常 结束 时 ,进程 的 状态 就 从 执行 态 转换 为 终止 态 。 

3. 七 状态 模型 

到 目前 为 止 ,总 是 假设 所 有 进程 都 在 主 存 中 。 事 实 上 ,为 了 更 好 地 管理 和 调度 进程 以 及 
适应 系统 的 功能 目标 ,引入 挂 起 状态 。 挂 起 进程 可 以 定义 为 暂时 被 淘汰 出 内 存 的 进程 (其 
PCB 仍然 保留 在 内 存 ) ,机 器 的 资源 是 有 限 的 ,在 资源 不 足 的 情况 下 ,操作 系统 对 在 内 存 中 
的 进程 进行 合理 的 安排 ,其 中 有 的 进程 被 暂时 调 离 出 内 存 , 当 条 件 允 许 的 时 候 ,会 被 操作 系 
统 再 次 调 回 内 存 。 具 有 挂 起 进程 功能 的 系统 中 的 进程 状态 ,两 个 新 状态 是 挂 起 就 绪 态 
(Ready Suspend) 和 挂 起 阻塞 态 (Blocked Suspend) ,如 图 2-12 所 示 。 


TO 请 求 


图 2-12 进程 的 7 种 状态 与 转换 


(1) 挂 起 就 绪 态 (Ready Suspend)。 进 程 具备 运行 条 件 , 但 目前 在 辅助 存储 器 中 ,只 有 
当 进 程 被 兑换 到 主 存 时 才能 调度 执行 。 

(2) 挂 起 阻塞 态 (Blocked Suspend)。 进 程 正在 等 待 某 一 事件 发 生 且 进程 在 辅助 存储 
器 中 。 

引起 进程 挂 起 的 原因 如 下 。 

(1) 终端 用 户 的 请 求 。 当 终端 用 户 在 自己 的 程序 运行 期 间 发 现 有 可 疑问 题 时 ,希望 暂 
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停 自己 的 程序 。 若 此 时 用 户 进程 正 处 于 就 绪 态 而 未 执行 , 则 该 进程 暂 不 接受 调度 ,以 便 用 户 
研究 其 执行 情况 或 对 程序 进行 修改 。 我 们 把 这 种 静止 状态 称 为 “ 挂 起 状态 ”。 

(2) 父 进程 的 请 求 。 有 时 父 进 程 希望 挂 起 自己 的 某 个 子 进程 ,以 便 考 察 和 修改 子 进程 ， 
或 者 协调 各 子 进程 间 的 活动 。 

(3) 负荷 调节 的 需要 。 当 实时 系统 中 的 工作 负荷 较 重 ,可 能 影响 到 对 实时 任务 的 控制 
时 ,可 由 系统 把 一 些 不 重要 的 进程 挂 起 ,以 保证 系统 能 正常 运行 。 

(4) 操作 系统 的 需要 。 操 作 系 统 有 时 希望 挂 起 某 些 进程 ,以 便 检 查 资源 使 用 情况 或 进 
行 记 账 。 

(5) 对 换 的 需要 。 为 了 缓和 内 存 紧 张 的 情况 ,将 内 存 中 处 于 阻塞 状态 的 进程 换 至 外 存 。 

如 果 一 个 进程 原来 处 于 执行 态 或 就 绪 态 ,此 时 可 因 挂 起 命令 而 由 原来 状态 变 为 挂 起 就 
绪 态 ,处 于 挂 起 就 绪 态 的 进程 不 能 参与 处 理 器 竞争 ; 当 处 于 挂 起 就 绪 状 态 的 进程 接 到 激活 
命令 后 , 它 就 由 原状 态 变 为 就 绪 态 。 具 体 可 有 以 下 几 种 情况 。 

(1) 阻塞 态 一 挂 起 阻塞 态 : 如 果 当 前 不 存在 就 绪 进 程 ,系统 根据 资源 分 配 状况 和 性 能 
要 求 ,选择 阻塞 态 进 程 对 换 出 去 ,使 之 处 于 挂 起 阻塞 态 。 

(2) 挂 起 阻塞 态 一 挂 起 就 绪 态 : 导致 进程 阻塞 的 事件 完成 后 ,相应 的 处 于 挂 起 阻塞 态 

(3) 挂 起 就 绪 态 一 就 绪 态 : 当主 存 中 不 存在 就 绪 进 程 ,或 者 挂 起 就 绪 态 进程 具有 比 就 
绪 态 进程 更 高 的 优先 级 ,系统 将 把 挂 起 就 绪 态 进程 换 回 主 存 并 转换 成 就 绪 态 。 

(4) 就 绪 态 一 挂 起 就 绪 态 : 系统 根据 当前 资源 分 配 状况 和 性 能 要 求 ,决定 把 就 绪 态 进 
程 对 换 出 去 ,使 之 成 为 挂 起 就 绪 态 。 

(5) 挂 起 阻塞 态 一 阻塞 态 : 进程 等 待 事件 发 生 时 ,原则 上 无 须 将 其 调 入 主 存 , 但 当 某 些 
进程 终止 后 ,系统 拥有 足够 的 自由 空间 ,而 某 个 挂 起 阻塞 态 进程 具有 较 高 的 优先 级 , 且 系 统 
得 知 导 致 其 阻塞 的 事件 即将 结束 , 便 可 能 发 生 这 一 类 状态 转换 。 

(6) 执行 态 一 挂 起 就 绪 态 : 当 一 个 具有 较 高 优先 级 的 挂 起 阻塞 进程 所 阻塞 的 事件 完成 
后 , 它 需 要 抢占 CPU, 而 此 时 主 存 空 间 不 足 ,可 能 会 导致 正在 运行 的 进程 转换 为 挂 起 就 绪 
态 。 另 外 ,执行 态 进程 也 可 自我 挂 起 。 

(7) 新 建 态 一 挂 起 就 绪 态 : 考虑 系统 当前 资源 分 配 状况 和 性 能 要 求 ,决定 将 新 进程 对 
换 出 去 ,使 之 处 于 挂 起 就 绪 态 。 

状态 的 转换 需要 原 语 来 实现 。 下 面 介绍 一 些 相 关 的 状态 转换 控制 原 语 。 

1. 进程 的 阻塞 

一 个 正在 运行 的 进程 ,因为 末 满 足 其 所 需求 的 资源 而 被 迫 处 于 阻塞 状态 ,等 待 所 需 事 件 
的 发 生 , 进 程 的 这 种 状态 的 变化 就 是 通过 进程 本 身 调用 阻塞 原 语 实现 的 。 阻 塞 是 进程 的 自 
主 行为 ,进程 阻塞 的 步骤 如 下 。 

步骤 1: 停止 进程 执行 ,将 现场 信息 保存 到 PCB。 
步骤 2: 修改 进程 PCB 的 有 关内 容 , 如 进程 状态 由 执行 态 改 为 阻塞 态 等 ,并 把 状态 已 修 
改 的 进程 移入 相应 事件 的 阻塞 队列 中 。 
步骤 3: 转 进 程 调度 程序 ,调度 其 他 进程 运行 。 

2. 进程 的 唤醒 
当 等 待 事件 完成 时 ,会 产生 一 个 中 断 , 激 活 操作 系统 ,在 系统 的 控制 之 下 将 被 阻塞 的 进 


程 唤醒 ,是 被 动 的 过 程 。 进 程 唤醒 的 步骤 如 下 。 
步骤 1: 从 相应 的 阻塞 队列 里 移出 进程 。 
步骤 2: 修改 进程 PCB 的 有 关内 容 , 如 进程 状态 改 为 就 绪 态 ,并 将 进程 移入 就 绪 队 列 。 
步骤 3: 若 被 唤醒 的 进程 比 当前 运行 进程 的 优先 级 高 ,重新 设置 调度 标志 。 
3. 进程 的 挂 起 
当 出 现 了 引起 进程 挂 起 的 事件 时 (例如 ,用 户 进程 请 求 将 自己 挂 起 ,或 父 进程 请 求 挂 起 
某 子 进程 ) 调 用 挂 起 原 语 , 进 程 挂 起 的 步骤 如 下 。 
步骤 1: 检查 被 挂 起 进程 的 状态 , 若 处 于 就 绪 态 , 便 将 其 改 为 挂 起 就 绪 ; 对 于 阻塞 态 的 
则 将 其 改 为 挂 起 阻塞 ,并 将 其 移入 到 指定 队列 。 
又 2: 把 该 进程 的 PCB 复制 到 某 指定 的 内 存 区 域 。 
步骤 3: 若 被 挂 起 的 进程 正在 执行 , 则 转向 调度 程序 重新 调度 。 
4. 进程 的 激活 
当 系 统 资源 充裕 或 请 求 激活 指定 进程 时 ,系统 或 相关 进程 会 调用 激活 原 语 把 指定 进程 
激活 ,激活 的 步骤 如 下 。 
步骤 1: 先 将 进程 从 外 存 调 入 内 存 。 
步骤 2: 检查 该 进程 的 现行 状态 ,若是 挂 起 就 绪 , 便 将 之 改 为 就 绪 ; 若 为 挂 起 阻塞 便 将 
之 改 为 阻塞 ,并 将 进程 移入 相应 队列 。 


2.2.5 进程 的 实现 


从 前 面 的 讲述 可 知 ,操作 系统 通过 进程 控制 块 (PCB) 感 知 进程 的 存在 ,为 了 实现 进程 模 
型 ,操作 系统 维护 着 一 张 表格 (一 个 结构 数组 ) , 即 进程 表 (Process Table) 。 每 个 进程 占 
个 进程 表 项 , 即 PCB。 

为 了 实现 进程 并 行 执行 的 错觉 ,操作 系统 需 不断 地 对 进程 切换 ,实质 上 ,为 了 完成 进程 
的 切换 ,还 需要 环境 的 支撑 ,如 硬件 寄存 器 ,程序 状态 字 寄 存 器 等 。 在 操作 系统 中 ,进程 物理 
实体 和 支持 进程 运行 的 环境 合成 进程 上 下 文 (Process Context) ,进程 在 当前 上 下 文中 运行 ， 
当 系 统 调度 新 进程 占有 处 理 器 时 ,新 老 进程 随 之 发 生 上 下 文 切 换 , 进 程 上 下 文 由 三 部 分 
组 成 。 

(1) 用 户 级 上 下 文 (User Level Context) 由 正文 (程序 ) 数据 .共享 存储 区 .用 户 栈 所 
组 成 。 

(2) 寄存 器 上 下 文 (Register Context) 由 程序 状态 字 寄 存 器 、 指 令 计数 器 、 栈 指针 (机 器 
状态 决定 它 指向 用 户 栈 或 核心 栈 ) 、 控 制 寄存 器 、 通 用 寄存 器 等 组 成 。 

(3) 系统 级 上 下 文 (System Level Context) 由 进程 控制 块 、 主 存 管理 信息 、 核 心 栈 等 
组 成 。 

当 进 程 的 切换 发 生 时 ,进程 在 执行 过 程 中 就 会 产生 中 断 , 将 控制 权 交 给 操作 系统 。 假 设 
当 一 个 磁盘 中 断 发 生 时 ,用 户 进程 正在 进行 , 则 中 断 硬件 将 程序 计数 器 .程序 状态 字 等 压 入 
堆栈 ,计算 机 随机 调转 到 中 断 向 量 所 指示 的 地 址 。 在 由 硬件 完成 上 述 操作 后 ,软件 ,特别 是 
中 断 服务 例 程 就 接管 一 切 剩余 的 工作 。 

所 有 的 中 断 都 从 保存 寄存 器 开始 ,对 于 当前 进程 而 言 ,通常 是 在 进程 表 项 中 。 随 后 ,会 
从 堆栈 中 删除 由 中 断 硬件 机 制 存 入 堆栈 的 那 部 分 信息 ,并 将 堆栈 指针 指向 一 个 由 进程 处 理 
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程序 所 使 用 的 临时 堆栈 。 通 常 , 该 例 程 可 以 供 所 有 的 中 断 使 用 ,因为 无 论 中 断 是 怎样 引起 
的 ,有 关 保 存 寄 存 器 的 工作 是 完全 一 样 的 。 

当 该 例 程 结束 后 , 它 调用 一 个 C 过 程 处 理 某 个 特定 的 中 断 类 型 剩 下 的 工作 (假设 操作 
系统 由 C 语言 编写 ) 。 在 完成 有 关 工 作 之 后 ,接着 调用 调度 程序 ,决定 随后 该 运行 哪个 进 
程 。 随 后 将 控制 权 转 给 一 段 汇 编 语 言 代码 ,为 当前 的 进程 装 入 寄存 器 值 以 及 内 存 映射 并 启 
动 该 进程 运行 。 中 断 处 理 和 调度 的 过 程 具体 如 下 。 

(1) 硬件 将 程序 计数 器 等 压 入 堆栈 。 

(2) 硬件 从 中 断 向 量 装 入 新 的 程序 计数 器 。 

(3) 汇编 语言 过 程 保存 寄存 器 值 。 

(4) 汇编 语言 过 程 设 置 新 的 堆栈 。 

(5) C 中 断 服 务 例 程 运行 (如 读 或 缓冲 输入 等 ) 。 

(6) 调度 程序 决定 下 一 个 将 运行 的 进程 。 

(7) C 过 程 返回 至 汇编 代码 。 

(8) 汇编 语言 过 程 开始 运行 新 的 当前 进程 。 


2.3 线 程 


线程 (Thread) 是 现代 操作 系统 中 出 现 的 一 个 重要 技术 ,目前 流行 的 操作 系统 几乎 都 采 
用 了 线程 机 制 。 线 程 的 引入 进一步 提供 了 程序 执行 的 并 发 性 ,提高 了 系统 的 效率 。 


2.3.1 线程 的 引入 及 定义 


在 传统 的 操作 系统 中 ,进程 是 系统 进行 资源 分 配 和 调度 的 基本 单位 ,以 进程 为 单位 分 配 
存放 其 映像 所 需要 的 虚 地 址 空间 ,执行 所 需要 的 主 存 空间 ,完成 任务 需要 的 其 他 各 类 外 围 设 
备 资源 和 文件 。 此 时 ,进程 是 调度 的 基本 单位 ,因而 在 创建 ,终止 和 切换 中 ,系统 必须 为 之 付 
出 较 大 的 时 空 开 销 。 为 了 减少 程序 并 发 执行 时 所 付出 的 时 空 开 销 , 使 得 并 发 粒度 更 细 , 并 发 
性 更 好 ,考虑 将 进程 的 两 项 功能 “独立 分 配 资源 "和 “被 调度 分 派 执行 ?分开 ,前 一 项 任务 仍 由 
进程 完成 ,后 一 项 任务 交 由 称 作 线 程 的 实体 来 完成 ,以 充分 发 挥 并 发 处 理 能 力 ,提高 系统 性 
能 。 在 这 种 思想 指导 下 ,形成 了 线程 的 概念 。 引 入 线程 还 有 一 个 好 处 ,就 是 能 较 好 地 支持 对 
称 多 处 理 器 系统 (Symmetric Multiprocessor,SMP)。 

所 谓 线程 ,是 进程 内 的 一 个 相对 独立 的 ,可 独立 调度 和 指派 的 执行 单元 ,是 进程 的 组 成 
部 分 。 线 程 的 组 成 部 分 有 : 

(1) 线程 的 唯一 标识 符 及 线程 状态 信息 , 即 线程 控制 块 (TCB) 。 

(2) 未 运行 时 所 保存 的 进程 上 下 文 ,可 把 线程 看 成 进程 中 一 个 独立 的 程序 计数 器 。 

(3) 核心 栈 , 在 核心 态 工 作 时 保存 参数 ,在 函数 调用 时 返回 地 址 等 。 

(4) 用 于 存放 线程 局 部 变量 和 用 户 栈 的 私有 存储 区 。 

传统 的 操作 系统 一 般 只 支持 单线 程 结构 进程 ,但 像 Windows、Solaris 等 很 多 操作 系统 
支持 多 线程 结构 进程 ,如 图 2-13 所 示 。 

由 于 线程 具有 许多 传统 进程 所 具有 的 特征 ,所 以 又 称 为 轻型 进程 (Light-Weight 
Process) ,相应 地 ,把 传统 进程 称 为 重型 进程 (Heavy-Weight Process)。 


E] 


单线 程 进程 (模型 ) 多 线程 进程 (模型 ) 


线程 线程 2 线程 i 线程 n 
进程 控 | | 用 户 堆 进程 控 | ”| 线程 控制 块 | | 线程 控制 块 | | 线程 控制 块 | “| 线程 控制 块 
制 块 制 块 

用 户 栈 用 户 栈 用 户 栈 
用 户 地 本 甩 
址 空间 | | 核心 栈 ee 核心 栈 核心 栈 核心 栈 


图 2-13 单线 程 结构 进程 和 多 线程 结构 进 


山 


2.3.2 线程 的 状态 


与 进程 一 样 ,线程 是 个 动态 的 概念 ,也 有 生命 周期 ,在 这 一 过 程 中 它 具 有 各 种 状态 ,虽然 
在 不 同 的 操作 系统 中 ,线程 的 状态 不 完全 相同 ,但 下 述 三 个 关键 的 状态 是 共有 的 。 

(1) 就 绪 状 态 : 线程 已 具备 执行 条 件 ,调度 程序 可 以 为 其 分 配 一 个 CPU 执行 。 

(2) 运行 状态 : 线程 正在 某 一 个 CPU 内 运行 。 

(3) 阻塞 状态 : 线程 正在 等 待 某 个 事件 发 生 , 则 被 阻塞 。 

线程 不 具有 进程 中 挂 起 状态 ,因为 挂 起 的 主要 作用 之 一 是 将 资源 从 内 存 移 到 外 存 , 而 线 
程 不 是 拥有 资源 的 基本 单位 , 它 不 应 该 有 将 整个 进程 或 线程 自己 从 主 存 移出 的 权限 。 同 时 ， 
需要 注意 的 是 当 一 个 线程 被 阻塞 后 ,为 了 保持 线程 的 灵活 性 和 优越 性 ,多 数 操作 系统 并 不 阻 
塞 整个 进程 ,该 进程 中 其 他 线程 依然 可 以 参与 调度 。 

线程 的 状态 转换 是 通过 相关 的 控制 原 语 实现 的 。 常 用 的 原 语 有 创建 线程 .终止 线程 、 线 
程 阻塞 等 。 创 建 线程 通常 被 称 为 派生 , 它 可 以 在 进程 内 派生 出 来 ,也 可 以 由 线程 派生 。 一 个 
新 派生 的 线程 具有 相应 的 数据 结构 指针 和 变量 ,然后 放 入 就 绪 队 列 。 如 果 一 个 线程 执行 完 
后 终止 该 线程 , 它 的 寄存 器 上 下 文 以 及 堆栈 内 容 将 被 释放 。 


2.3.3 线程 的 特征 


根据 线程 的 概念 ,线程 具有 以 下 一 些 特征 。 

(1) 线程 是 进程 中 的 一 个 相对 可 独立 运行 的 单元 。 

(2) 线程 是 操作 系统 中 的 基本 调度 单位 ,在 线程 中 包含 调度 所 需要 的 基本 信息 。 

(3) 在 具备 线程 机 制 的 操作 系统 中 ,进程 不 再 是 调度 单位 ,一 个 进程 中 至 少 包含 一 个 线 
程 ,以 线程 作为 调度 单位 。 

(4) 线程 自己 并 不 拥有 资源 , 它 与 同 进程 中 的 其 他 线程 共享 该 进程 所 拥有 的 资源 。 由 
于 线程 之 间 涉 及 资源 共享 ,所 以 需要 同步 机 制 来 实现 进程 内 多 线程 之 间 的 通信 。 

(5) 与 进程 类 似 , 线 程 还 可 以 创建 其 他 线程 ,线程 也 有 生命 周期 ,也 有 状态 的 变化 。 

线程 具有 许多 类 似 于 进程 的 特征 ,我 们 将 比较 线程 与 进程 ,从 中 更 清楚 地 了 解 线 程 所 具 
有 的 特征 。 

(1) 拥有 资源 方面 。 不 管 是 以 进程 为 基本 单位 的 操作 系统 ,还 是 在 引进 线程 的 操作 系 


进程 与 线程 


媳 吉 


所 作 系 统 原 理 


统 中 ,进程 都 是 独立 拥有 资源 的 一 个 基本 单位 。 它 可 以 申请 并 拥有 自己 的 资源 ,也 可 以 访问 
其 所 属 进 程 的 资源 。 而 线程 只 拥有 一 点 儿 在 运行 中 必要 的 资源 ,不 过 它 可 以 访问 所 属 进程 
的 资源 ,但 资源 仍然 是 分 给 进程 的 。 
(2) 调度 方面 。 引 入 线程 后 ,进程 是 资源 的 拥有 者 ,线程 是 处 理 机 调度 和 分 配 的 单位 。 
因此 在 同一 进程 内 ,线程 的 切换 不 会 引起 进程 的 切换 ,而 由 一 个 进程 中 的 线程 切换 到 另 一 个 
进程 中 的 线程 时 则 需要 引起 进程 间 的 切换 。 

(3) 并 发 性 方面 。 引 入 线程 后 ,不 仅 进程 之 间 可 以 并 发 执行 ,而 且 一 个 进程 内 的 多 个 线 
程 之 间 也 可 以 并 发 执行 ,因此 系统 具有 了 更 好 的 并 发 性 ,进而 提高 了 系统 的 资源 利用 率 和 香 
吐 量 。 

(4) 系统 开销 方面 。 进 程 切换 时 有 很 大 的 时 空 开 销 ,而 线程 切换 时 只 需要 保 在 和 设置 
少量 的 寄存 器 ,时 空 开 销 很 小 。 另 外 ,由 同一 进程 内 的 多 个 线程 共享 进程 的 同一 地 址 空间 ， 
因此 ,多 个 线程 之 间 的 同步 与 通信 也 非常 容易 实现 ,甚至 不 需要 操作 系统 内 核 的 干预 。 


2.3.4 线程 的 分 类 


在 不 同 的 操作 系统 中 ,线程 的 实现 方法 不 完全 相同 。 可 以 将 多 线程 的 实现 方法 分 成 三 
类 : 用 户 级 线程 .内核 支持 线程 和 混合 式 线程 ,如 图 2-14 所 示 。 无 论 什么 类 型 的 线程 ,都 必 
须 以 直接 或 间接 方式 获得 操作 系统 内 核 支持 ,内 核 支持 线程 可 以 直接 使 用 系统 调用 为 它 服 
务 , 而 用 户 级 线程 若 要 取得 内 核 服 务 , 则 必须 借助 于 一 个 中 间 系 统 。 下 面 分 别 对 几 种 类 型 加 
以 介绍 。 


(a) 用 户 级 线程 (b) 内 核 支持 线程 (e) 混合 式 线程 


ar @ kr (®) Process | 


图 2-14 三 种 线程 实现 方式 


1. 用 户 级 线程 

用 户 级 线程 (User-Level Thread,ULT) 由 用 户 程序 创建 ,并 由 用 户 程序 对 其 进行 调度 
和 管理 。 这 种 线程 的 创建 .终止 .切换 以 及 通信 都 不 能 直接 利用 系统 调用 完成 ,而 是 借助 线 
程 库 这 个 中 间 系 统 实现 。 线 程 库 是 操作 系统 提供 的 一 个 专门 用 来 管理 用 户 级 线程 的 软件 
包 , 它 驻 留 在 用 户 空间 内 ,提供 了 创建 线程 .终止 线程 、 线 程 切换 、 线 程 调度 .线程 同步 以 及 线 
程 之 间 通 信 等 功能 。 线 程 库 在 用 户 级 线程 与 内 核 之 间 起 到 了 接口 作用 。 用 户 级 线程 通过 线 
程 库 以 间接 方式 获得 内 核 提供 的 服务 ,从 而 使 用 户 级 线程 与 内 核 无 关 ; 反 过 来 ,由 于 线程 库 


的 隔离 ,内 核 也 不 知道 用 户 级 线程 的 存在 。 

这 种 方法 具有 以 下 优点 。 

(1) 用 户 级 线程 的 切换 无 须 通 过 陷入 (内 中 断 ) 进 入 内 核 ,切换 操作 在 进程 的 用 户 空间 
中 进行 ,用 于 管理 线程 的 数据 结构 均 保 存在 进程 的 用 户 空间 内 ,因此 用 户 级 线程 的 切换 速度 
高 于 内 核 支 持 线 程 的 切换 速度 ,而 且 系 统 开销 小 。 

(2) 用 户 级 线程 可 以 运行 在 任何 操作 系统 上 ,就 是 在 不 支持 线程 的 操作 系统 上 也 可 以 
实现 用 户 级 线程 。 

(3) 由 于 线程 调度 由 线程 库 实现 ,而 线程 库 的 线程 调度 算法 与 系统 的 进程 调度 算法 无 
关 , 因 此 线程 调度 灵活 方便 。 各 应 用 程序 可 以 根据 需要 在 线程 库 中 选择 不 同 的 线程 调度 算 
法 ,而 不 会 干扰 内 核 的 进程 调度 程序 。 

这 种 方法 具有 以 下 缺点 。 

(1) 由 于 内 核 不 知道 用 户 级 线程 的 存在 ,因此 , 当 用 户 级 线程 执行 一 个 系统 调用 时 , 系 
统 将 阻塞 该 线程 所 属 的 整个 进程 ,使 得 该 进程 内 的 所 有 线程 都 不 能 运行 ,从 而 降低 了 线程 的 

(2) 在 ULT 方式 下 ,多 线程 不 便利 用 多 处 理 器 ,因为 每 次 只 有 一 个 进程 的 一 个 线程 在 

一 个 CR 上 运 和 

2. 内 核 级 线程 

内 核 级 线程 (Kernel-Level Thread,KLT) 中 ,所 有 线程 的 创建 .调度 .管理 都 由 操作 系 
统 内 核 负 责 。 一 个 用 户 进程 可 以 按 多 线程 方式 编写 程序 , 当 它 被 提交 给 多 线程 操作 系统 运 
行 时 ,内 核 会 为 它 创 建 一 个 进程 和 一 个 线程 ,线程 在 运行 中 还 可 以 创建 新 的 线程 。 每 当 创建 
一 个 新 线程 时 ,操作 系统 内 核 就 在 内 核 空 间 为 该 线程 分 配 一 个 线程 控制 块 TCB, 用 来 登记 
该 线程 的 线程 标识 符 、 寄 存 器 内 容 、 状 态 以 及 优先 级 等 信息 ,并 分 配 运 行 所 必需 的 资源 。 每 
当 撤 销 一 个 线程 时 ,内 核 便 回 收 为 该 线程 分 配 的 资源 和 线程 控制 块 。 由 此 可 见 , 内 核 支持 线 
程 的 创建 和 终止 类 似 于 传统 进程 的 创建 与 终止 。 内 核 支持 线程 的 调度 和 切换 也 与 传统 进程 
的 调度 和 切换 类 似 , 由 内 核 完 成 。 

这 种 方法 具有 以 下 优点 。 

(1) 在 引入 内 核 支持 线程 的 操作 系统 中 ,调度 以 线程 为 单位 ,内 核能 够 同时 调度 一 个 进 
程 内 的 多 个 线程 并 发 执行 ; 而 在 多 处 理 器 系统 中 , 则 能 同时 将 多 个 线程 分 配 到 各 个 处 理 器 
上 并 行 执行 ; 因此 ,内 核 支持 线程 比较 适合 多 处 理 器 系统 。 

(2) 在 引入 内 核 支持 线程 的 操作 系统 中 ,一 个 线程 因 等 待 某 个 事件 而 阻塞 不 会 影响 其 
他 线程 执行 。 

(3) 内 核 支持 线程 本 身 只 使 用 了 很 小 的 数据 结构 和 堆栈 ,切换 速度 较 快 ,加 之 内 核 本 身 
也 可 以 采用 多 线程 技术 实现 ,因此 ,引入 内 核 支持 线程 的 操作 系统 一 般 具 有 较 高 的 运行 
效率 。 

这 种 方法 的 缺点 是 线程 运行 在 用 户 态 ,而 线程 的 调度 和 管理 由 内 核实 现 , 以 至 于 同一 进 
程 中 的 线程 需要 在 用 户 态 和 核心 态 之 间 来 回 切换 ,系统 开销 较 大 。 

3. 混合 式 线程 

混合 式 线程 将 上 述 用 户 级 与 内 核 级 结合 。 在 这 种 方式 下 ,一 方面 内 核 支持 多 线程 的 创 
建 . 调 度 和 管理 等 操作 , 另 一 方面 系统 为 用 户 提供 一 个 线程 库 , 允许 用 户 程序 建立 \` 调 度 和 管 
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理 用 户 级 线程 。 
2.3.5 多 核 和 多 线程 


通过 上 述 介绍 ,每 个 单位 时 间 内 ,CPU 只 能 处 理 一 个 线程 ,以 这 样 的 单位 进行 ,如 果 想 
要 在 单位 时 间 内 处 理 超 过 一 个 的 线程 , 是 不 可 能 的 。 英特尔 通过 超 线程 技术 
(Hyperthreading, HT) 成 为 第 一 家 实现 在 一 个 实体 处 理 器 中 提供 两 个 逻辑 线程 的 公司 。 英 
特 尔 的 HT 技术 是 以 单个 核心 处 理 单元 ,去 整合 两 个 逻辑 处 理 单元 ,也 就 是 一 个 实体 核心 ， 
两 个 逻辑 核心 。 近 似 地 说 , 超 线程 允许 CPU 保持 两 个 不 同 的 线程 状态 ,然后 在 纳 秒 级 的 时 
间 尺 度 内 来 切换 。 超 线程 不 提供 真正 的 并 行 处 理 。 在 一 个 时 刻 只 有 一 个 进程 在 运行 ,但 是 
线程 的 切换 时 间 则 减少 到 纳 秒 级 数量 ,因此 可 以 在 单位 时 间 内 处 理 两 个 线程 ,模拟 双核 心 运 
作 。 对 于 软件 来 说 ,要 想 实现 HT 技术 ,就 需要 创建 多 个 线程 。 这 种 技术 对 操作 系统 而 言 
是 有 意义 的 , 因为 每 个 线程 在 操作 系统 看 来 就 像 是 单个 的 CPU。 考 虑 一 个 实际 有 两 个 
CPU 的 系统 ,每 个 CPU 有 两 个 线程 ,这 样 操作 系统 将 把 它 看 成 是 4 个 CPU。 可 以 看 出 , 单 
核 CPU 运用 超 线程 技术 可 以 大 致 模拟 出 双核 的 效果 ,双核 CPU 运用 超 线程 技术 可 以 大 致 
模拟 出 4 核 的 效果 ,这 就 很 好 地 解释 了 我 们 常 听 说 的 双核 4 线程 或 4 核 8 线程 ; 而 不 管 
CPU 是 单 核 , 双 核 还 是 具备 超 线程 技术 的 单 核 ,双核 ,都 可 以 实现 多 线程 编程 。 

前 面 提 到 了 单 核 ,双核 。 那 么 什么 是 多 核 呢 ?” 所 谓 多 核 是 将 两 个 或 多 个 完整 的 CPU， 
通常 称 为 核 (Core), 放 到 同一 芯片 上 (技术 上 来 说 是 同一 小 硅 片 ), 如 图 2-15 所 示 。CPU 可 
能 共享 高 速 缓存 也 可 能 不 共享 ,但 它们 都 共享 内 存 。 


LI y 如 
缓存 < 了 
核 1 核 2 
多 四 
核 3 核 4 


(a) 带 有 共享 L2 缓 存 的 4 核 芯片 (b) 带 有 分 离 L2 缓 存 的 4 核 蕊 片 
图 2-15 4 核 芯片 


这 样 的 设计 结果 是 多 核 芯片 就 相当 于 小 的 多 处 理 机 。 实 际 上 ,多 核 芯片 时 常 被 称 为 片 
级 多 处 理 机 (Chip-level MultiProcessors, CMP)。 从 软件 的 角度 来 看 ,CMP 与 基于 总 线 的 
多 处 理 机 和 使 用 交换 网 络 的 多 处 理 机 并 没有 太 大 的 差别 。 不 过 它们 还 是 存在 着 若干 的 差 
别 。 例 如 ,基于 总 线 的 多 处 理 机 ,每 个 CPU 拥有 自己 的 高 速 缓存 , 因 特 尔 使 用 的 共享 高 速 
缓存 的 设计 并 没有 出 现在 其 他 的 多 处 理 器 中 。CMP 与 其 他 多 处 理 器 的 另 一 个 差异 是 容错 。 
因为 CPU 之 间 的 连接 非常 紧密 ,一 个 共享 模块 的 失效 可 能 导致 许多 CPU 同时 出 错 。 而 这 
样 的 情况 在 传统 的 多 处 理 机 中 是 很 少 出 现 的 。 


2.4 处理 器 调度 


在 多 道 程序 环境 下 ,通常 会 有 多 个 进程 或 线程 竞争 CPU。 这 就 要 求 系统 能 按 某 种 算 
法 ,选择 下 一 个 要 运行 的 进程 。 在 操作 系统 中 完成 选择 工作 的 那 一 部 分 称 为 调度 程序 
(Scheduler) ,该 程序 使 用 的 算法 称 为 调度 算法 (Scheduling Elgorithm)。 


2.4.1 调度 的 功能 与 时 机 


现代 操作 系统 中 ,按照 调度 所 实现 的 功能 来 分 ,通常 把 处 理 器 分 配给 进程 或 线程 的 调度 
称 为 低级 调度 , 除 此 之 外 ,还 有 中 级 调度 和 高 级 调度 ,它们 一 起 构成 三 级 调度 体系 。 其 中 , 低 
级 调度 是 该 体系 中 不 可 缺少 的 最 基本 调度 。 下 面 将 具体 介绍 这 三 级 调度 的 功能 及 其 调度 的 
时 机 。 

1. 高 级 调度 

在 了 解 高 级 调度 之 前 , 先 来 了 解 一 个 概念 , 即 作业 。 作 业 是 一 个 比 程序 更 广泛 的 概念 ， 
它 不 仅 包含 通常 的 程序 和 数据 ,而 且 还 应 配 有 一 份 作业 说 明 书 ,系统 根据 该 说 明 书 来 对 程序 
的 运行 进行 控制 。 在 批 处 理 系 统 中 ,是 以 作业 为 基本 单位 从 外 存 调 入 内 存 的。 作业 通常 分 
成 若干 个 既 相 对 独立 又 互相 关联 的 加 工 步骤 ,每 个 步骤 称 为 一 个 作业 步 。 每 个 作业 步 可 能 
对 应 一 个 或 多 个 进程 。 例 如 ,一 个 用 Java 语言 编写 的 程序 可 看 作 一 个 作业 ,该 作业 执行 时 ， 
首先 经 过 JDK 编译 程序 进行 编译 ,形成 后 级 名 为 class 的 字 节 码 文件 ; 字 节 码 文件 再 通过 
JDK 的 执行 程序 进行 解释 执行 ,用 户 才能 看 到 最 终 运行 结果 。 对 上 述 两 个 步骤 ,系统 可 以 
通过 创建 两 个 进程 来 完成 。 如 系统 此 时 还 有 其 他 进程 运行 ,这 两 个 进程 与 其 他 进程 并 发 
执行 。 

作业 一 般 要 经 历 "提交 ”“ 后 备 ”“ 执 行 " 和 “完成 ”4 个 状态 ,如 图 2-16 所 示 。 用 户 向 系 
统 提交 一 个 作业 时 ,该 作业 所 处 的 状态 为 提交 状态 。 例 如 ,将 一 套 作 业 卡 片 交 给 机 房管 理 
员 , 由 管理 员 将 它们 放 到 读 卡 机 上 读 入 ; 或 者 用 户 通过 键盘 向 机 器 输入 作业 等 。 用 户 作 业 
经 输入 设备 (如 读 卡 机 ) 送 入 输入 井 ( 磁 盘 的 一 部 分 ) ,等 待 进 入 内 存 时 所 处 状态 为 后 备 状 态 。 
后 备 态 作 业 的 数据 已 经 转换 成 机 器 可 读 的 形式 ,作业 请 求 资源 等 信息 也 交 给 了 操作 系统 。 
系统 中 往往 有 多 个 作业 处 于 后 备 状态 ,它们 通常 被 组 织 成 队列 形式 。 后 备 态 作业 被 作业 调 
度 程序 选中 后 调 入 内 存 , 获 得 所 需 资源 且 正 在 处 理 机 上 执行 时 , 称 作 业 处 于 执行 状态 。 作 业 
执行 完毕 ,其 结果 被 放 到 硬盘 中 专门 用 来 存放 结果 的 某 个 固定 区 域 或 打印 输出 ,系统 收回 分 
配给 它 的 全 部 资源 ,此 时 的 作业 处 于 完成 状态 。 
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图 2-16 作业 状态 及 其 转换 
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在 大 型 通用 系统 中 ,往往 有 数 百 个 终端 与 主机 相连 ,共用 系统 中 的 一 台 主 机 。 某 一 时 
刻 , 系 统 中 可 能 有 数 百 个 作业 放 在 磁盘 的 批 处 理 作业 队列 中 。 如 何 从 这 些 作业 中 选 出 作业 
放 入 内 存 是 处 理 机 调度 的 重要 功能 之 一 。 

高 级 调度 (High Level Scheduling) 又 称 作 业 调 度 或 长 程 调 度 , 它 是 根据 某 种 算法 将 外 
存 上 处 于 后 备 作业 队列 中 的 若干 作业 调 入 内 存 , 为 作业 分 配 所 需 资源 并 建立 相应 进程 。 

高 级 调度 决定 允许 哪些 作业 可 进入 内 存 , 参 与 竞争 CPU 和 系统 其 他 资源 ,将 一 个 或 一 
批 作业 从 后 备 状 态 变 为 执行 状态 。 被 高 级 调度 程序 选中 的 作业 可 获得 基本 内 存 和 相应 的 系 
统 资源 ,系统 为 之 创建 相应 的 进程 。 此 后 ,该 作业 以 进程 的 形式 参与 并 发 执行 , 同 其 他 进程 
竞争 CPU。 高 级 调度 为 中 级 调度 和 低级 调度 做 好 了 前 期 准备 。 

在 多 道 批 处 理 系统 中 ,为 了 管理 和 调度 作业 ,系统 为 每 个 作业 设置 了 一 个 作业 控制 块 
(Job Control Block,JCB) , 它 记 录 作 业 的 相关 信息 。 作 业 控 制 块 是 作业 存在 的 标志 ,只 有 作 
业 执 行 完 成 或 中 途 退 出 系统 时 ,作业 控制 块 才 被 撤销 。 在 JCB 中 所 包含 的 内 容 因 系统 而 
异 , 通 常 应 包含 的 内 容 有 : 作业 标识 ,用户 名 称 . 用 户 账号 .作业 类 型 .作业 状态 .调度 信 
息 .资源 需求 .进入 系统 时 间 、. 开 始 处 理 时 间 、 作 业 完 成 时 间 、 作 业 退 出 时 间 ,资源 使 用 情 
况 等 。 

2. 中 级 调度 

中 级 调度 (Intermediate Level Scheduling) 又 称 内 存 调度 , 它 是 进程 在 内 存 和 外 存 之 间 

的 对 换 , 引 入 中 级 调度 的 目的 是 为 了 提高 内 存 利用 率 和 系统 吞吐 量 ,控制 系 统 并 发 度 ,降低 
系统 开销 。 当 内 存 空间 非常 紧张 或 处 理 机 无 法 找到 一 个 可 执行 的 就 绪 进 程 时 , 需 把 某 些 暂 
时 不 能 运行 的 进程 换 到 外 存 上 去 等 待 ,释放 出 其 占用 的 宝贵 内 存 资源 给 其 他 进程 使 用 。 换 
到 外 存 的 进程 所 处 状态 为 挂 起 状态 。 当 这 些 进程 重新 具备 运行 条 件 且 内 存 又 有 空闲 空间 
时 ,由 中 级 调度 程序 决定 把 外 存 上 的 某 些 进程 重新 调 入 内 存 , 并 修改 其 状态 ,为 占用 处 理 器 
做 好 准备 。 
中 级 调度 实际 上 是 存储 管理 中 的 对 换 功能 , 它 控制 进程 对 主 存 的 使 用 。 在 虚拟 存储 管 
理 系统 中 ,进程 只 有 被 中 级 调度 选中 , 才 有 资格 占用 主 存 。 中 级 调度 可 以 控制 进程 对 主 存 的 
使 用 ,从 某 种 意义 上 讲 , 中 级 调度 可 通过 设 定 内 存 中 能 够 接纳 的 进程 数 来 平衡 系统 负载 ,在 
一 定时 间 内 起 到 平衡 和 调整 系统 负载 的 作用 。 

3. 低级 调度 

低级 调度 (Low Level Scheduling) 又 称 进程 调度 .短程 调度 , 它 决 定 哪个 就 绪 态 进程 获 
得 处 理 器 , 即 选择 某 个 进程 从 就 绪 态 变 为 执行 态 。 执 行 低级 调度 的 原因 多 是 处 于 执行 态 的 
进程 由 于 某 种 原因 放弃 或 被 剥夺 处 理 机 。 

进程 调度 的 功能 主要 包括 以 下 两 部 分 。 

(1) 选择 就 绪 进程 。 动 态 查找 就 绪 状 态 进程 队列 中 各 进程 的 优先 级 和 资源 (主要 是 内 
存 ) 使 用 情况 ,按照 一 定 的 进程 调度 算法 确定 处 理 机 的 分 配对 象 。 

(2) 进程 切换 。 进 程 切 换 是 处 理 器 分 配 的 具体 实施 过 程 。 正 在 处 理 器 上 执行 的 进程 释 
放 处 理 器 ,将 调度 程序 选中 的 就 绪 态 进程 切换 到 处 理 器 上 执行 。 进 程 切换 中 主要 完成 的 工 
作 有 : 保存 当前 被 切换 进程 的 执行 现场 ,累计 当前 就 绪 进 程 的 执行 时 间 、 剩 余 时 间 片 .动态 
变化 优先 级 等 。 调 度 程序 根据 进程 调度 策略 选择 一 个 就 绪 态 进程 ,把 其 状态 转换 为 执行 态 ， 
并 把 处 理 器 分 配给 它 。 进 程 的 执行 现场 往往 保存 在 自己 的 PCB、 用 户 栈 和 系统 栈 中 , 常 包 


括 以 下 寄存 器 内 容 : 处 理 器 状态 寄存 器 ,指令 地 址 寄存 器 、 通 用 寄存 器 、 堆 栈 起 始 地 址 和 栈 
顶 指针 存储 管理 寄存 器 。 

进程 调度 是 最 基本 的 一 种 调度 , 它 可 以 采用 非 抢 占 方式 和 抢占 方式 。 

(1) 非 抢 占 方式 。 在 这 种 方式 下 ,一 旦 分 派 程 序 把 处 理 机 分 配给 某 个 进程 后 , 便 让 它 一 
直 运行 下 去 ,直到 进程 完成 或 发 生 某 事件 而 阻塞 时 , 才 把 处 理 机 分 配给 另 一 进程 。 即 使 在 就 
绪 队 列 中 存在 优先 级 高 于 当前 执行 进程 的 进程 ,当前 进程 仍 将 占用 处 理 机 直到 该 进程 自己 
因 调用 原 语 操作 或 等 待 /O 而 进入 阻塞 状态 ,或 时 间 片 用 完 时 才 重 新 发 生 调度 让 出 处 
理 机 。 

这 种 调度 方式 的 优点 是 实现 简单 、 系 统 开销 小 ,适用 于 大 多 数 的 批 处 理 系 统 环 境 。 但 它 
难以 满足 紧急 任务 的 要 求 一 一 立即 执行 ,因而 可 能 造成 难以 预料 的 后 果 。 显 然 ,在 要 求 比较 
严格 的 实时 系统 中 ,不 宜 采 用 这 种 调度 方式 。 

(2) 抢占 方式 。 在 这 种 方式 下 , 某 个 进程 正在 运行 时 可 以 被 系统 以 某 种 原则 剥夺 已 分 
配给 它 的 处 理 器 ,将 处 理 器 分 配给 其 他 进程 。 即 就 绪 队列 中 一 旦 有 优先 级 高 于 当前 执行 进 
程 优先 级 的 进程 存在 时 , 便 立即 发 生 进程 调度 ,转让 处 理 器 。 在 这 种 调度 方式 中 ,进程 调度 
程序 可 根据 某 种 原则 停止 正在 执行 的 进程 ,将 已 分 配给 当前 进程 的 处 理 器 收回 ,重新 分 配给 
另 一 个 处 于 就 绪 状 态 的 进程 。 

抢占 方式 的 优点 是 可 以 防止 一 个 长 进程 长 时 间 占 用 处 理 器 ,能 为 大 多 数 进程 提供 更 公 
平 的 服务 ,特别 是 能 满足 对 响应 时 间 有 着 严格 要 求 的 实时 任务 的 需求 。 但 抢占 方式 调度 所 
需 付 出 的 开销 较 大 。 

在 上 述 三 级 调度 中 ,低级 调度 是 各 类 操作 系统 必 备 的 功能 ,在 纯粹 的 分 时 操作 系统 或 实 
时 操作 系统 中 ,通常 不 需要 高 级 调度 ; 一 般 的 操作 系统 都 配置 高 级 调度 和 低级 调度 ; 而 功 
能 完善 的 操作 系统 为 了 提高 主 存 的 利用 率 和 作业 吞吐 量 , 引 入 了 中 级 调度 。 可 见 不 同 的 操 
作 系 统 采 用 的 调度 方式 是 不 同 的 ,调度 程序 的 优化 也 是 不 同 的 ,根据 不 同 的 环境 ,划分 出 三 
种 , 即 批 处 理 , 交 互 式 和 实时 。 我 们 也 将 在 介绍 完 调度 目标 后 ,针对 这 三 种 环境 对 调度 算法 
做 详细 的 介绍 。 


2.4.2 调度 算法 的 目标 


操作 系统 调度 算法 的 选择 会 受到 很 多 因素 的 影响 ,评价 调度 算法 的 优 劣 和 性 能 也 是 十 
分 复杂 的 事情 ,不 同类 型 操作 系统 的 调度 算法 往往 不 一 样 ,由 于 应 用 进程 的 特性 ,对 响应 时 
间 和 系统 资源 的 要 求 不 尽 相 同 , 使 得 调度 算法 的 设计 比较 复杂 。 通 常情 况 下 如 何 选择 调度 
方式 和 算法 ,很 大 程度 上 取决 于 操作 系统 的 类 型 及 其 目标 。 选 择 调度 算法 时 需要 考虑 用 户 
的 要 求 ,同时 也 应 该 考虑 系统 的 总 体 性 能 。 

1. 面向 用 户 

(1) 周转 时 间 短 , 即 用 户 等 待 输出 的 时 间 短 。 周 转 时间 是 指 从 作业 提交 给 系统 到 作业 
完成 为 止 的 这 段 时 间 间 隔 。 周 转 时 间 包 括 4 部 分 : 作业 在 外 存 后 备 队 列 上 等 待 (作业 ) 调 度 
的 时 间 , 进 程 在 就 绪 队 列 上 等 待 进程 调度 的 时 间 , 进 程 在 CPU 上 执行 的 时 间 , 以 及 进程 等 
待 1/O 操作 完成 的 时 间 。 一 个 作业 的 周转 时 间 等 于 作业 的 完成 时 间 减 到 达 时 间 , 或 者 是 执 
行 时 间 加 等 待 时 间 。 

每 个 用 户 都 希望 自己 作业 的 周转 时 间 尽 可 能 的 短 。 但 是 对 于 一 个 计算 机 系统 而 言 ,总 
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是 希望 能 使 所 有 任务 的 平均 周转 时 间 最 短 。 平 均 周 转 时 间 的 定义 为 
T= >| 


nL i= 

其 中 ,n 为 作业 数 ,T; 为 第 i 个 作业 周转 时 间 。 
在 周转 时 间 里 ,一 个 执行 时 间 长 的 作业 和 执行 时 间 短 的 作业 去 直接 比较 周转 时 间 是 不 
公平 的 ,所 以 提出 了 带 权 周转 时 间 , 即 作业 的 周转 时 间 了 与 作业 的 执行 时 间 了 , 之 比 : Wi; 二 


Ti/Ts 称 为 作业 i 的 带 权 周转 时 间 , 而 平均 带 权 周转 时 间 可 表示 为 


W= 1Bw| 

(2) 响应 时 间 快 , 即 用 户 交 互 快捷 。 响 应 时 间 是 指 从 提交 一 个 请 求 到 开始 接受 响应 之 
间 的 这 段 时 间 。 它 通常 用 于 评价 分 时 系统 和 实时 系统 。 它 包括 三 部 分 时 间 : 从 键盘 输入 的 
请 求 信息 传送 到 处 理 器 的 时 间 , 处 理 器 对 请 求 信息 进行 处 理 的 时 间 , 以 及 将 所 形成 的 响应 信 
息 回 送 到 终端 显示 器 的 时 间 。 

(3) 截止 时 间 保 证 。 截 止 时 间 是 指 某 任务 必须 开始 执行 的 最 迟 时间 , 或 必须 完成 的 最 
迟 时 间 。 对 于 实时 系统 而 言 ,为 了 确保 任务 能 在 有 限 的 时 间 范 围 内 做 出 正确 的 反馈 ,必须 要 
求 任务 在 截止 时 间 之 前 开始 执行 ,否则 将 可 能 造成 难以 预料 的 后 果 。 

(4) 优先 权 。 在 批 处 理 , 分 时 和 实时 系统 中 选择 调度 算法 时 ,可 以 将 急需 处 理 的 作业 设 
置 成 较 高 的 优先 权 , 然 后 根据 优先 权 高 低 选择 调度 ,使 得 优先 权 高 的 作业 能 得 到 及 时 的 处 
理 。 在 实时 系统 中 ,有 时 为 了 确保 作业 能 被 及 时 处 理 , 还 需要 选择 抢占 式 的 调度 方式 。 

2. 面向 系统 

(1) 资源 利用 率 高 。 使 得 CPU 或 其 他 资源 能 够 并 行 工作 ,并 且 资 源 的 利用 率 尽 可 能 
高 。 由 于 CPU 价格 昂贵 ,致使 CPU 的 利用 率 成 为 衡量 系统 性 能 的 十 分 重要 的 指标 。 

CPU 的 利用 率 王 CPU 有 效 工 作 时 间 /CPU 总 的 运行 时 间 
CPU 总 的 运行 时 间 王 CPU 有 效 工作 时 间 十 CPU 空闲 等 待 时 间 

(2) 系统 吞吐 量 高 。 吞 吐 量 是 单位 时 间 内 处 理 的 作业 数 。 这 是 选择 批 处 理 系 统 性 能 的 
一 个 重要 指标 。 对 于 大 型 作业 ,一 般 吞 吐 量 为 每 小 时 一 道 作 业 , 对 于 中 小 型 作业 ,其 吞吐 量 
则 可 能 达到 每 小 时 数 十 道 作 业 之 多 。 作 业 调 度 算法 的 选择 对 吞吐 量 的 影响 很 大 ,对 于 同一 
批 作业 ,车 采用 较 好 的 调度 方式 , 则 可 显著 地 提高 系统 的 吞吐 量 。 

(3) 公平 性 。 确 保 每 个 用 户 每 个 进程 获得 合理 的 CPU 份额 或 其 他 资源 份额 ,不 会 出 现 
饿 死 的 情况 。 


2.4.3 批 处 理 作业 调度 


在 这 一 节 中 ,我 们 将 考查 在 批 处 理 系统 中 作业 调度 的 算法 。 

1. 先 来 先 服务 

先 来 先 服务 (First-Come First-Served,FCFS) 调 度 算法 是 最 简单 的 一 种 调度 算法 , 它 不 
仅 可 以 用 于 高 级 调度 ,也 可 以 用 于 低级 调度 。 当 在 作业 调度 中 采用 该 算法 时 ,每 次 从 作业 后 
备 队 列 中 选择 一 个 等 待 时 间 最 长 的 作业 调 入 内 存 ,并 为 其 分 配 资源 ,建立 进程 ,然后 放 入 就 
绪 队列 。 

这 是 一 种 非 剥 夺 式 调度 算法 ,易于 实现 ,但 效率 不 高 。 只 顾及 作业 的 等 候 时 间 , 不 考虑 


作业 要 求 服务 时 间 的 长 短 , 不 利于 短 作 业 而 优待 长 作业 。 有 时 为 了 等 待 长 作业 执行 结束 , 短 
作业 的 周转 时 间 和 带 权 周转 时 间 将 变 得 很 大 ,从 而 使 若干 作业 的 平均 周转 时 间 和 平均 带 权 
周转 时 间 也 变 得 很 大 。 下 面 通过 一 个 例子 来 分 析 一 下 FCFS 调度 算法 。 

例 2-1 表 2-2 给 出 了 5 个 作业 到 达 系 统 的 时 间 、 运 行 时 间 , 给 出 作业 的 调度 顺序 ,计算 
各 自 的 周转 时 间 和 带 权 周转 时 间 ,平均 周转 时 间 和 平均 带 权 周转 时 间 。 


表 2-2 作业 情况 
作业 号 A B G 蕊 E 
到 达 时 间 0 1 2 3 4 
服务 时 间 4 3 5 2 4 
解 : 
slalecln|lel| 

0 4 第 12 14 18 

作业 的 调度 顺序 : A 一 BC 一 DE 


业 的 周转 时 间 : 
Ta=4—0=4 Ts=7—1=6 Tc=12—2=10 
Tp=14—3=11 Tre=18—4=14 
作业 的 带 权 周 转 时 间 : 
Wa=4/4=1 Was=6/3=2 We=10/5=2 
Wop=11/2=5.5 Ws=14/4=3.5 
作业 的 平均 周转 时 间 : 


T= 
作业 的 平均 带 权 周转 时 间 : 
WH [WatWatWetWotWe]—14/5—2.8 

2. 短 作 业 优 先 调度 算法 

短 作 业 优先 调度 算法 (Short Job First,SJF) 是 以 进入 系统 的 作业 所 要 求 的 CPU 运行 
时 间 的 长 短 为 标准 ,总 是 选取 预计 计算 时 间 最 短 的 作业 优先 调度 的 算法 。 其 从 后 备 队列 中 
选择 一 个 或 若干 个 估计 运行 时 间 最 短 的 作业 ,将 它们 调 入 内 存 运行 。 

短 作业 优先 调度 算法 是 一 种 非 抢占 式 的 调度 算法 ,能 够 克服 FCFS 算法 的 缺点 ,易于 实 
现 ,但 执行 效率 不 高 。 短 作业 优先 调度 算法 存在 不 容 忽视 的 缺点 。 

(1) 该 算法 对 长 作业 不 利 。 更 严重 的 是 ,如 果 有 一 长 作业 进入 系统 的 后 备 队列 ,由 于 调 
度 程序 总 是 优先 调度 那些 (即使 是 后 进来 的 ) 短 作业 ,进入 系统 早 的 长 作业 长 期 得 不 到 处 理 ， 
将 导致 饥饿 现象 。 

(2) 该 算法 完全 未 考虑 作业 的 紧迫 程度 ,因而 不 能 保证 紧迫 性 作业 会 被 及 时 处 理 。 

(3) 由 于 作业 的 长 短 只 是 根据 用 户 所 提供 的 估计 执行 时 间 而 定 的 ,而 用 户 又 可 能 会 有 
意 或 无 意 地 缩短 其 作业 的 估计 运行 时 间 , 致 使 该 算法 不 一 定 能 真正 做 到 短 作 业 优 先 调度 。 

下 面 通过 一 个 例子 来 分 析 一 下 SJF 调度 算法 。 


二 [Ta+ Tat+Tct+To+Te]=9 
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例 2-2 按照 SJF 算法 给 出 表 2-2 中 作业 的 执行 顺序 ,计算 各 自 的 周转 时 间 和 带 权 
时 间 , 平 均 周转 时 间 和 平均 带 权 周转 时 间 。 


A| 人 DI|B| 和 BIEl 


| 
4 6 9 13 18 


此 的 调度 顺序 : A 一 D 一 BE 一 C 

: 业 的 周转 时 间 : 
TaA=4—0=4 Te 一 9 一 1 一 8 Tc=18—2=16 
To 一 6 一 3 一 3 Te 一 13 一 4 一 9 

作业 的 带 权 周转 时 间 : 

WaA=4/4=1 Ws=8/3=2.67 We = 16/5= 3.2 
Wop=3/2=1.5 Ws= 9/4= 2.25 

作业 的 平均 周转 时 间 : 


bp 3 [Ti TFTe+ T+ Te]=8 
作业 的 平均 带 权 周转 时 间 : 
WwW 二 [WA 十 Wa 十 We 十 Wo 十 We] 


3. 高 响应 比 调度 算法 

高 响应 比 优先 (Highest Response Ratio Next, HRRN) 调 度 算法 是 对 FCFS 调度 算法 
和 短 作业 优先 调度 算法 的 一 种 综合 平衡 。FCFS 算法 只 考虑 等 待 时 间 而 未 考虑 运行 时 间 的 
长 短 , 短 作业 优先 调度 算法 只 考虑 运行 时 间 而 未 考虑 等 待 时 间 的 长 短 。 因 此 这 两 种 调度 算 
法 在 某 些 情况 下 都 有 不 足 之 处 。 高 响应 比 优先 调度 算法 中 的 优先 权 的 变化 规律 可 描述 为 


， 等 待 时 间 十 要求 服务 时 间 
优先 家 要 求 服务 时 间 
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从 上 面 的 公式 可 以 看 出 : 

(1) 如 果 作业 的 等 待 时 间 相 同 , 则 要 求 服务 的 时 间 越 短 , 其 优先 权 越 高 ,因此 该 算法 在 
等 待 时 间 相 同 的 作业 中 会 选择 短 作 业 , 有 利于 短 作 业 。 

(2) 当 要 求 服务 的 时 间 相 同时 ,作业 的 优先 权 决 定 于 其 等 待 时 间 , 等 待 时间 越 长 ,其 优先 
权 越 高 ,因此 对 运行 时 间 相 同 的 作业 该 算法 会 选择 等 待 时 间 长 的 作业 , 即 类 似 于 先 来 先 服务 。 

(3) 对 于 长 作业 ,作业 的 优先 级 可 以 随 等 待 时间 的 增加 而 提高 , 当 其 等 待 时 间 足 够 长 时 ， 
其 优先 级 便 可 升 到 很 高 ,从 而 也 可 获得 处 理 机 。 因 此 对 长 作业 而 言 , 不 会 出 现 “ 饥 饿 ”现象 。 

总 之 ,该 算法 既 照顾 了 短 作 业 , 又 考虑 了 作业 到 达 的 先后 次 序 , 不 会 使 长 作业 长 期 得 不 
到 服务 。 

例 2-3 按照 HRRN 算法 给 出 表 2-2 作业 的 执行 顺序 ,计算 各 自 的 周转 时 间 和 带 权 周 
转 时 间 , 平 均 周 转 时 间 和 平均 带 权 周 转 时 间 。 

解 : 开始 时 只 有 作业 A, 作 业 A 被 选中 ,执行 时 间 4; 作业 A 执行 完毕 后 ,B,C,D,E 的 
响应 比 依次 为 5/3,7/5,3/2,4/4, 作 业 B 被 选中 ,执行 时 间 3; 作业 B 执行 完毕 后 ,C,D,E 
的 响应 比 依 次 为 10/5,6/2,7/4, 作 业 DD 被 选中 ,执行 时 间 2; 作业 D 执行 完毕 后 ,C,E 的 响应 


比 依次 为 12/5,9/4, 作 业 C 被 选中 ,执行 时 间 5; 最 后 作业 王 被 选中 ,执行 时 间 4。 所 以 有 
^|ese|lplclel 
4 7 9 14 18 


作业 的 调度 顺序 : A 一 BD 一 CE 
作业 的 周转 时 间 : 
TA=4—0=4 Ts=7—1=6 Tc=14—2=12 
Tp=9—3=6 Te=18—4=14 
作业 的 带 权 周转 时 间 : 
Wa = 4/4 Ws=6/3=2 We= 12/5= 2.4 
Wbp=6/2=3 We= 14/4= 3.5 
作业 的 平均 周转 时 间 : 
42 


一 二 [TA+ Tat Te+ To+Te] 一 名 一 8.4 
作业 的 平均 带 权 周转 时 间 : 
11.9 


亚 二 [WA Was 十 We 十 Wp 十 We 一 一 一 2. 38 


批 处 理 系统 中 除了 有 作业 调度 外 ,还 有 进程 调度 ,上 述 三 种 方法 除了 适用 于 作业 调度 ， 
也 适用 于 进程 调度 ,调度 算法 的 原理 基本 相同 ,同时 某 些 算法 既 可 以 用 在 批 处 理 系统 中 ,也 
可 以 用 在 交互 式 系统 中 。 如 短 作 业 优 先 常 常 伴随 着 最 短 响应 时 间 , 短 作业 优先 调度 如 果 是 
进程 的 话 ,那么 其 短 进程 优先 调度 也 可 适用 于 交互 式 系统 。 


2.4.4 交互 系统 进程 调度 


现在 考察 用 于 交互 式 系统 中 的 一 些 进程 调度 算法 ,他 们 在 个 人 计算 机 、 服 务 器 和 其 他 类 
系统 中 都 是 常用 的 。 

1. 时 间 片 轮转 调度 算法 

时 间 片 轮转 法 调度 (Round Robin,RR) 也 称 为 时 间 片 调度 或 轮转 调度 ,是 分 时 系统 中 
采用 的 调度 算法 ,其 基本 思想 是 为 每 一 个 进程 分 配 一 个 时 间 段 ,该 时 间 段 被 称 为 时 间 片 ,上 
允许 该 进程 运行 的 时 间 。 通 常情 况 下 时 间 片 的 大 小 为 几 十 到 几 百 毫秒 。 每 个 进程 只 能 依次 
循环 轮流 运行 ,如 果 时 间 片 结束 时 进程 还 在 运行 ,CPU 将 剥夺 该 进程 的 使 用 权 转 而 将 CPU 
分 配给 另 一 个 进程 。 如 果 进 程 在 时 间 片 结束 之 前 阻塞 或 结束 ,CPU 当即 进行 切换 。 为 了 实 
现 进程 的 循环 执行 ,将 每 次 被 中 止 运行 的 进程 存 入 就 绪 队 列 的 末尾 ,同时 将 CPU 分 配给 就 
绪 队 列 中 的 队 首 进程 。 该 算法 是 一 种 简单 而 又 公平 的 算法 ,使 用 非常 广泛 。 

时 间 片 轮转 算法 是 一 种 抢占 式 调度 算法 。 使 用 这 种 算法 进行 调度 ,系统 耗费 在 进程 ( 线 
程 ) 切 换 上 的 开销 比较 大 ,而 开销 的 大 小 与 时 间 片 的 长 短 有 很 大 的 关系 。 若 时 间 片 太 短 , 则 
大 多 数 进程 (线程 ) 都 不 可 能 在 一 个 时 间 片 内 运行 结束 ,于 是 频繁 切换 ,系统 开销 显著 增 大 。 
反之 ,车 时 间 片 过 长 ,长 到 所 有 进程 (线程 ) 都 可 以 在 一 个 时 间 片 内 运行 结束 , 则 该 算法 退化 
成 先 来 先 服务 算法 ,在 这 种 情况 下 ,用 户 对 响应 时 间 的 要 求 将 得 不 到 保证 。 因 此 ,为 了 使 用 
户 的 输入 能 够 得 到 及 时 响应 ,同时 又 不 会 增加 太 多 系统 开销 ,时 间 片 的 选择 应 与 完成 一 个 基 
本 交互 过 程 所 需要 的 时 间 相 当 , 从 而 保证 大 部 分 基本 交互 过 程 都 能 在 一 个 时 间 片 内 完成 。 
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例 2-4 当时 间 片 是 1 和 4 时 ,按照 RR 算法 给 出 如 表 2-3 所 示 进 程 的 执行 顺序 ,计算 
各 自 的 周转 时 间 和 带 权 周转 时 间 , 平 均 周 转 时 间 和 平均 带 权 周转 时 间 。 


表 2-3 进程 情况 
进程 号 A B wo D E 
到 达 时 间 0 i 间 4 
服务 时 间 4 3 5 2 4 


解 : (1) 时 间 片 为 1 时 
进程 的 执行 顺序 : 

A|B|IA|I|CcIB|IDIA|IE|IC|IBIDPIAIE|IC|IE|IC|IEICI| 
1 入 3 4 过 6 下 8 9 10 11 12 对 14 15 16 17 18 

进程 的 周转 时 间 : 

T= 

二 

进程 的 带 权 周转 时 间 : 

Wa =12/4=3 Ws=9/3=3 We=16/5=3.2 

Wi 

进程 的 平均 周转 时 间 : 

过 了 CTA | 

进程 的 平均 带 权 周转 时 间 : 

= 二 [WA 十 We 十 We 十 Wan 十 We 一 


co 


5 


也 :看 


a 


16. 45 


=29 


(2) 时 间 片 为 4 时 
进程 的 执行 顺序 : 
A| slcelp|lelel., 
A HW 


进程 的 周转 时 间 : 


Top=13—3=10 Ts=17—4=13 
进程 的 带 权 周转 时 间 : 

Wa=4/4=1 Ws=6/3=2 We= 16/5= 3.2 
Wo =10/2=5 Ws= 13/4= 3,25 


进程 的 平均 周转 时 间 : 
T= 击 [Ta+Ts+Te+To 二 Te] 一 委 一 9.8 
进程 的 平均 带 权 周转 时 间 : 
W— 吉 [WatWat WetWotWe]— ls —2. 89 


2. 优先 级 调度 算法 

高 优先 级 调度 算法 的 基本 思想 是 : 当 发 生 调 度 时 ,总 是 调度 当前 处 于 就 绪 队 列 中 优 
先 级 最 高 的 进程 ,使 之 获得 处 理 机 。 在 这 种 算法 中 存在 两 种 方式 ; 非 抢占 式 优 先 权 调度 
算法 和 抢占 式 优先 权 调度 算法 。 而 根据 优先 权 的 类 型 ,又 可 分 为 静态 优先 级 和 动态 优 

非 抢 占 式 优先 级 调度 方式 规定 系统 如 果 把 处 理 机 分 配给 就 绪 队 列 中 优先 权 最 高 的 进程 
后 ,该 进程 将 持续 获得 CPU 的 使 用 权 , 直 到 进程 完成 或 发 生 某 事件 而 阻塞 时 , 才 把 处 理 机 
分 配给 另 一 个 进程 。 这 种 调度 方式 的 优点 是 实现 简单 、 系 统 开销 小 ,适用 于 大 多 数 的 批 处 理 
系统 。 

抢占 式 优先 级 调度 方式 规定 首先 把 处 理 机 分 配给 优先 权 最 高 的 进程 ,使 该 进程 占用 
CPU 执行 。 但 在 其 执行 期 间 , 如 果 系 统 中 进入 了 一 个 优先 权 更 高 的 进程 ,进程 调度 程序 就 
立即 停止 当前 进程 ( 原 优 先 权 最 高 的 进程 ) 的 执行 ,重新 将 处 理 嚣 分 配给 新 到 的 优先 级 最 高 
的 进程 。 这 种 调度 方式 的 优点 是 能 更 好 地 满足 紧迫 作业 的 要 求 ,因此 适用 于 要 求 比较 严格 
的 实时 系统 ,以 及 对 性 能 要 求 较 高 的 批 处 理 和 分 时 系统 。 

在 优先 级 调度 算法 中 ,进程 的 优先 级 一 般 由 优先 数 决定 。 
静态 优先 数 是 指 进程 在 创建 时 就 获得 一 个 整数 数值 ,此 数值 在 进程 的 整个 运行 过 程 中 

定 不 变 。 优 先 数 的 大 小 反映 进程 优先 级 的 高 低 。 有 的 系统 规定 越 大 的 优先 数 其 优先 级 越 
高 ,当然 也 可 以 反 过 来 规定 。 优 先 数 的 决定 一 般 取决 于 进程 类 型 ,资源 需求 量 ,紧迫 程度 入 
用 户 需求 等 。 

动态 优先 数 是 指 进程 的 优先 级 随 着 进程 的 推进 可 以 动态 改变 。 现 代 操 作 系 统 中 ,采用 
优先 级 调度 算法 的 系统 大 多 使 用 的 是 动态 优先 数 的 策略 。 动 态 优先 数 的 选择 可 以 根据 进程 
占有 CPU 的 时 间 长 短 以 及 就 绪 进 程 等 待 CPU 的 时 间 长 短 来 确定 。 

例 2-5 表 2-4 给 出 5 个 进程 到 达 就 绪 队 列 时 的 时 间 、 执 行 时 间 和 优先 数 ,优先 数 越 大 
优先 级 越 高 ,按照 优先 级 调度 算法 采用 抢占 方式 给 出 进程 的 执行 顺序 ,计算 各 自 的 周转 时 间 
和 带 权 周 转 时 间 , 平 均 周 转 时 间 和 平均 带 权 周转 时 间 。 


表 2-4 进程 情况 
进程 号 A B 如 D E 
到 达 时 间 0 1 区 3 4 
服务 时 间 4 3 6 2 4 
优先 数 1 2 4 5 5 


解 : 进程 的 执行 顺序 : 


sleleslelsleltl 
1 -1 


进程 的 周转 时 间 : 
Ta=18—0=18 Ts=15—1=14 Tc=11—2=9 


进程 与 线程 


媳 吉 


握 作 系统 原理 


进程 的 带 权 周 转 时 间 : 
WaA=18/4=4.5 Ws=14/3=4.67 Wc = 9/5=1.8 
Wp=10/2=5 Wse=4/4=1 
进程 的 平均 周转 时 间 : 
T=#$ [Tt Tet TetTot Te]=—11 


进程 的 平均 带 权 周转 时 间 : 
W=H [Wat WeatWetWo+We]— 
3. 多 级 反馈 队列 调度 算法 
多 级 反馈 队列 调度 (Multi-Level Feedback Queue,MLFQ) 又 被 称 为 反馈 循环 队列 ,如 
图 2-17 所 示 。 不 必 事 先知 道 各 种 进程 所 需 的 执行 时 间 、 优 先 数 等 ,而 且 可 以 满足 不 同类 型 
进程 的 需求 。 它 采用 动态 分 配 优先 数 , 调 度 策 略 是 一 种 抢占 式 的 调度 方法 。 该 算法 是 目前 
公认 的 较 好 的 一 种 进程 调度 算法 。 其 主要 思想 是 由 系统 建立 多 个 就 绪 队 列 , 每 个 队列 对 应 
于 一 个 优先 级 ,第 一 个 队列 的 优先 级 最 高 ,第 二 个 队列 的 优先 级 次 之 ,其 后 队列 的 优先 级 逐 
个 降低 。 较 高 优先 级 队列 的 进程 /线程 分 配给 较 短 的 时 间 片 , 较 低 优先 级 队列 的 进程 /线程 
分 配给 较 长 的 时 间 片 ,最 后 一 个 队列 进程 /线程 按 FCFS 算法 进行 调度 。 当 一 个 新 进程 进入 
内 存 后 首先 将 它 放 入 第 一 队列 的 末尾 , 当 轮 到 该 进程 执行 时 ,如 果 它 能 在 该 时 间 片 内 完成 ， 
便 可 准备 撤离 系统 ,如 果 它 在 一 个 时 间 片 结束 时 尚未 完成 ,调度 程序 便 将 该 进程 转 入 第 二 队 
列 的 末尾 ,如 果 它 在 第 二 队列 中 运行 一 个 时 间 片 后 仍 未 完成 ,再 依次 将 它 放 入 第 三 队列 …… 
如 此 下 去 。 处 理 器 调度 每 次 先 从 第 一 个 队列 中 选取 执行 者 ,同一 队列 中 的 进程 /线程 按 
FCFS 原则 排队 ,只 有 在 未 选 到 时 , 才 从 较 低 一 级 的 就 绪 队 列 中 选取 , 仅 当前 面 所 有 队列 为 
空 时 , 才 会 运行 最 后 一 个 就 绪 队 列 中 的 进程 /线程 。 
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Si 至 


就 结 队 列 1 CpU 
| Wl cpu 
一 一 一 一 至 
[La | cpu 


一 就 绪 队列 CPU 


时 间 片 :Si<Sy<Sy<-…<S， 


图 2-17 多 级 反馈 队列 


多 级 反馈 队列 调度 算法 是 一 种 性 能 较 好 的 作业 低级 调度 策略 ,能 够 满足 各 类 用 户 的 需 
要 。 根 据 用 户 运 行 作业 的 类 型 进行 分 析 , 可 以 发 现 : 
(1) 终端 型 用 户 ( 分 时 交互 型 ) 所 提交 的 大 多 属于 交互 型 ,这 种 类 型 的 进程 通常 比较 小 ， 
系统 只 要 能 使 这 些 进程 在 第 一 队列 所 规定 的 时 间 片 内 完成 ,就 可 以 使 终端 型 用 户 感到 满意 。 
(2) 短 批 处 理 进程 只 需 仅 在 第 一 队列 或 在 第 一 、 第 二 队列 中 各 执行 一 个 时 间 片 就 能 完 


成 工作 ,周转 时 间 仍然 很 短 。 


(3) 假如 一 个 长 进程 进入 内 存 , 它 最 终 必 将 移入 优先 级 最 低 的 就 绪 队 列 中 ,如 果 其 后 有 


源源 不 断 的 短 进程 进入 内 存 , 则 长 进程 一 直 等 待 ,陷入 “饥饿 ”状态 。 解 决 此 问题 的 一 种 方法 
是 对 于 低 优 先 级 队列 总 等 待 时 间 足 够 长 的 进程 提升 优先 级 ,从 而 让 它 获 得 运行 机 会 。 

例 2-6 设 系统 中 有 三 个 就 绪 队 列 , 第 一 个 队列 的 时 间 片 为 1, 第 二 个 队列 的 时 间 片 为 
2, 第 三 个 队列 的 时 间 片 为 4。 按 照 MLFQ 算法 给 出 表 2-3 进程 的 执行 顺序 ,计算 各 自 的 周 
转 时 间 和 带 权 周转 时 间 , 平 均 周转 时 间 和 平均 带 权 周 转 时 间 。 

解 : 进程 的 执行 顺序 : 


A | B 


下 县 于: 测 医 : 避 荐 : 汪 攻 "可 且 攻 项: 有 本 二 本 < 本 : 才 | 


1 2 EE 4 El 7 9 了 一 |” Oy i | 


进程 的 周转 时 间 : 


Ta 


15—0=15 Ts=9—1=8 Tc=17—2=15 


Tp 


Wa = 


12 一 3 一 9 TE 一 18 一 4 一 14 


进程 的 带 权 周转 时 间 : 


15/4=3.75 Ws=8/3=2.67 Wc = 15/5=3 


Wo 


9/2 一 4.5 WE 一 14/4 一 3.5 


WwW 


进程 的 平均 周转 时 间 : 


YG 二 CTA 十 Ta 十 Te 十 To 十 Te] 5 


进程 的 平均 带 权 周转 时 间 ， 
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二 [WA 十 Wa 十 We 十 Wo 十 We] 3.48 


2.4.5 实时 系统 进程 调度 


实时 系统 是 一 种 时 间 起 主导 作用 的 系统 ,对 时 间 有 着 严格 的 要 求 。 在 实时 系统 中 ,每 一 
个 实时 任务 都 有 一 个 时 间 约 束 要 求 。 实 时 调度 (Real-time Scheduling) 的 目标 就 是 合理 地 
安排 这 些 任 务 的 执行 次 序 , 使 之 满足 各 个 实时 任务 的 时 间 约 束 要 求 。 

通常 ,一 个 特定 任务 与 一 个 截止 时 间 相 关联 。 截 止 时 间 包 括 : 开始 截止 时 间 ( 任 务 在 某 


时 间 以 前 必须 开始 执行 ) 和 完成 截止 时 间 ( 任 务 在 某 时 间 以 前 必须 完成 ) 。 


实时 调度 策略 主要 考虑 如 何 使 硬 实时 任务 在 规定 的 截止 时 间 内 完成 (或 开始 ) 。 同 时 ， 
尽 可 能 使 软 实时 任务 也 能 在 规定 的 截止 时 间 内 完成 (或 开始 )。 周 转 时 间 和 香 吐 量 等 则 不 再 
显得 重要 。 大 多 数 现代 操作 系统 都 无 法 实现 直接 依据 任务 截止 时 间 进 行 调度 ,它们 一 般 通 


过 提高 响应 速度 保证 外 


E 务 在 其 要 求 的 截止 时 间 内 完成 。 因 此 ,实时 调度 应 具备 如 下 的 基本 


信息 ,才能 安排 好 合适 的 实时 调度 策略 。 
(1) 就 绪 时 间 。 它 是 指 任务 成 为 就 绪 态 所 需 的 时 间 。 


(2) 开始 截止 时 间 


和 完成 截止 时 间 。 通 常 不 需要 两 者 都 知道 ,典型 的 实时 系统 只 需 知 


道 任 务 的 开始 截止 时 间或 完成 截止 时 间 。 


(3) 任务 的 执行 时 


间 。 


(4) 实时 任务 执行 时 的 资源 需求 。 


进程 与 线程 


其 久 避 


握 作 系统 原理 


(5) 实时 任务 的 优先 级 ,通常 硬 实时 任务 的 优先 级 较 高 。 

(6) 子 任务 结构 。 一 个 较 大 的 任务 可 以 分 解 成 一 个 必须 执行 的 子 任务 和 若干 个 可 选 的 
子 任 务 。 

一 般 情况 下 ,实时 系统 中 可 能 同时 有 多 个 周期 性 任务 并 发 执行 ,形成 任务 流 , 这 些 实时 
任务 都 要 求 系统 做 出 实时 响应 。 系 统 能 否 对 它们 全 部 予以 处 理 ,取决 于 每 个 任务 要 求 的 处 
理 时 间 和 该 任务 出 现 的 周期 。 例 如 ,系统 中 有 m 个 周期 性 任务 ,其 中 任务 i 出现 的 周期 为 


P,, 处 理 所 需 CPU 时 间 为 Ci, 那 么 系统 能 处 理 这 个 任务 流 的 条 件 是 > 6: < 1。 
i=1 i 


当 此 值 等 于 1 时 ,处 理 器 利用 率 达 到 最 大 。 这 是 实时 调度 的 理想 状态 ,但 实际 中 往往 比 
此 值 小 。 满 足 这 个 不 等 式 关系 的 实时 系统 称 为 可 调度 。 该 式 称 为 可 调度 测试 公式 。 

当 此 值 大 于 1 时 ,实时 调度 算法 失效 。 无 论 进行 何 种 调度 都 不 能 满足 实时 要 求 。 此 时 ， 
采取 的 方法 主要 有 : 减少 任务 流 中 的 周期 性 任务 数量 m, 系统 更 换 性 能 更 好 的 CPU ,减少 每 
个 实时 任务 所 需 的 CPU 时 间 C;, 增 加 系统 中 处 理 器 的 个 数 ,提高 系统 的 处 理 能 力 等 。 

举例 来 说 ,一 个 实时 系统 要 处 理 三 个 任务 ,其 出 现 周 期 分 别 为 1 00ms、200ms 和 500ms， 
如 果 任 务 的 处 理 时 间 分 别 为 50ms、30ms 和 100ms, 则 这 个 系统 是 任务 可 调度 的 ,因为 0.5 十 
0.15 十 0.2 达 1。 如 果 加 入 周期 为 1s 的 第 4 个 任务 ,只 要 其 处 理 时 间 不 超过 150ms, 此 实时 
系统 仍 将 是 任务 可 调度 的 。 当 然 , 隐 含 条 件 是 进程 切换 的 时 间 足 够 短 , 可 以 忽略 不 计 。 

下 面 介 绍 一 个 典型 的 实时 调度 算法 一 一 最 早 截止 时 间 优 先 调度 算法 (Earliest Deadline 
First,EDF) 。 

EDF 根据 实时 任务 的 开始 截止 时 间 确 定 任 务 的 优先 级 ,截止 时 间 越 早 , 其 优先 级 越 高 。 
调度 程序 把 所 有 可 以 运行 的 进程 按照 其 截止 时 间 先 后 顺序 放 在 一 个 以 表格 形式 存在 的 就 绪 
队列 中 , 队 首 的 任务 具有 最 早 截止 时 间 。 调 度 程序 运行 时 ,总 是 选 队 首 进程 。 

对 于 新 到 达 的 实时 任务 ,系统 查看 其 截止 时 间 。 如 果 截 止 时 间 先 于 正在 运行 任务 的 截 
止 时 间 ,新 进程 抢占 当前 进程 的 CPU 使 用 权 。 最 早 截止 时 间 优 先 调度 算法 是 抢占 式 调度 
算法 ,适用 于 周期 性 和 非 周 期 性 实时 任务 的 调度 。 

例 2-7 系统 中 有 三 个 实时 任务 A、B、C， 表 2-5 进程 情况 
进程 的 执行 时 间 和 周期 如 表 2-5 所 示 。 如 果 这 
三 个 进程 都 在 :一 0 时 刻 就 绪 , 在 每 个 周期 都 会 


实时 任务 ”所 需 执行 时 间 周期 


在 截止 时 间 产 生 一 个 实例 ,例如 ,任务 A 在 0、 ee ee 
30.60 时 间 点 会 产生 实例 A1、A2 和 A3。 并 且 c Se Soms 
每 个 周期 的 截止 时 间 如 图 2-18 所 示 , 用 EDF 
算法 给 出 它们 的 执行 过 程 。 


Al 截 BI 截 Cl 截 A2 截 B2 截 A3 截 C2 截 A4.B3 截 
oi Li i 四 加 上 加 地 四 | 最 加 


30 40 50 60 80 90 10 120 
图 2-18 各 任务 各 周期 的 截止 时 间 


解 : 在 时 刻 0 时 ,有 任务 实例 A1、Bl 和 C1, 根据 截止 时 间 分 别 是 30、40、50。 按 EDF 
原则 ,A 先 运行 , 当 到 时 刻 10 时 ,B 运行 ,运行 结束 后 C 运行 ,在 C 运行 结束 后 ,到 达 时 刻 


30, 这 时 A 产生 实例 A2, 此 时 就 绪 队 列 只 有 任务 A. 所 以 A 运行 ,以 此 类 推 ,各 任务 的 执行 
过 程 如 图 2-19 所 示 。 


AL|BL|cl|Az|B2|cz|A3| |B3|A4|cs| |As|B4| 
10 25 30 40 55 60 70 80 95 105 110 120 130 145 时 间 (ms) 


图 2-19 最 早 截止 时 间 优 先 调 度 算法 


2.4.6 线程 调度 


支持 线程 技术 的 操作 系统 中 存在 两 个 层面 的 并 发 活动 : 进程 并 发 和 线程 并 发 。 在 这 样 
的 系统 中 ,线程 是 低级 调度 的 基本 单位 ,线程 调度 与 线程 实现 方式 关系 密切 。 本 节 主 要 介绍 
日 户 级 线程 调度 和 核心 级 线程 调度 。 

1. 用 户 级 线程 调度 

用 户 级 线程 是 在 用 户 态 下 创建 的 。 系 统 内 核 并 不 知道 线程 的 存在 。 此 时 系统 内 核 还 是 
和 以 前 一 样 的 操作 ,只 为 进程 服务 ,从 就 绪 进 程 队列 中 选中 一 个 进程 并 分 配给 它 一 个 CPU 
时 间 片 。 假 设 该 进程 为 A, 进 程 A 内 部 的 线程 调度 程序 决定 该 进程 中 哪个 线程 运行 。 假 设 
获得 CPU 时 间 片 的 线程 为 Al ,由 于 并 发 执行 的 同一 进程 内 的 多 个 线程 之 间 不 存在 时 钟 中 
断 , 故 线程 Al 执行 时 不 受 时 钟 中 断 的 干扰 。 如 果 线 程 Al 用 完了 进程 A 的 时 间 片 ,系统 内 
核 就 会 调度 另 一 个 进程 执行 。 当 进程 A 再 次 获得 时 间 片 时 ,线程 Al 将 恢复 运行 。 如 此 反 
复 ,直到 Al 完成 自己 的 工作 。 如 果 线 程 Al 运行 时 间 较 短 , 没 用 完 一 个 时 间 片 就 已 结束 或 
被 强行 终止 ,线程 Al 让 出 CPU ,进程 A 的 线程 调度 程序 调度 进程 A 的 另 一 个 线程 运行 , 例 
如 线程 A2。 

具体 线程 调度 算法 可 采用 典型 进程 调度 算法 。 从 实用 角度 考虑 ,时 间 片 轮转 调度 和 优 
先 级 调度 更 为 有 效 。 用 户 级 线程 调度 的 局 限 是 缺乏 时 钟 中 断 及 时 将 运行 时 间 过 长 的 线程 中 
断 , 不 能 照顾 短线 程 。 

2. 核心 级 线程 调度 

在 核心 支持 线程 技术 的 系统 中 ,内 核 直 接 调度 线程 。 线 程 调度 时 ,内 核 不 考虑 该 线程 属 
于 哪个 进程 。 被 选中 的 线程 获得 一 个 时 间 片 ,如 果 执 行 时 间 超 过 此 时 间 片 ,该 线程 被 系统 强 
制 挂 起 。 如 果 线程 在 给 定 的 时 间 片 内 阻塞 ,处 于 内 核 的 线程 调度 程序 调度 男 一 个 线程 运行 。 
后 者 和 前 者 可 能 同属 于 一 个 进程 ,也 可 能 属于 不 同 进程 。 

假设 进程 A 的 线程 Al 获得 一 个 长 度 为 30ms 的 时 间 片 ,5ms 之 后 该 线程 被 阻塞 ,让 出 
CPU 使 用 权 。 此 时 ,内 核 调度 程序 把 CPU 分 配给 其 他 线程 ,可 能 分 给 进程 A 的 线程 ,也 可 
能 分 给 进程 B 的 线程 ,出 现 属于 不 同 进程 间 的 线程 切换 。 

用 户 级 线程 调度 和 核心 级 线程 调度 的 主要 区 别 如 下 。 

(1) 用 户 级 线程 间 切 换 只 需 少 量 机 器 指令 ,速度 较 快 ; 而 核心 级 线程 间 切 换 需 要 完整 
的 进程 上 下 文 切换 ,修改 内 存 映 像 ,高 速 缓存 失效 ,因而 速度 慢 ,系统 开销 大 。 

(2) 用 户 级 线程 可 使 用 专 为 某 用 户 态 程序 定制 的 线程 调度 程序 ,应 用 定制 的 线程 调度 
程序 能 够 比 内 核 更 好 地 满足 用 户 态 程序 需要 。 核 心 级 线程 在 内 核 中 完成 线程 调度 ,内 核 不 
了 解 每 个 线程 的 作用 ,不 能 做 到 这 一 点 。 


进程 与 线程 


其 久 避 
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小 结 


一 口 


进程 是 操作 系统 中 最 重要 和 最 基本 的 概念 之 一 ,引入 进程 是 系统 资源 的 有 限 性 和 操作 
系统 内 操作 的 并 发 性 所 决定 的 。 进 程 具有 生命 周期 ,由 创建 而 产生 ,由 调度 而 执行 ,由 终止 


而 消亡 ， 


操作 系统 的 基本 功能 是 进程 的 创建 .管理 和 终止 。 为 了 实现 对 进程 的 管理 ,每 个 进 


程 有 唯一 标志 一 一 进程 控制 块 ,创建 进程 必须 为 其 创建 进程 控制 块 ,终止 进程 时 系统 便 回收 


进程 控制 


| 块 。 


如 果 说 操作 系统 中 引入 进程 的 目的 是 使 多 个 进程 并 发 执行 ,以 便 改 善 资源 利用 率 和 提 
高 系统 效率 ,那么 ,在 操作 系统 中 引入 线程 则 是 为 了 减少 程序 并 发 执行 时 所 付出 的 时 空 代 


价 , 线 程 


的 实现 有 用 户 级 线程 .内 核 级 线程 和 混合 式 三 种 方式 。 


处 理 器 调度 分 为 三 级 : 高 级 调度 、 中 级 调度 和 低级 调度 。 根 据 不 同 的 调度 目标 有 一 大 
批 研究 出 来 的 调度 算法 。 有 些 算法 主要 用 于 批 处 理 系统 ,有 些 算法 常用 在 批 处 理 系统 和 交 
互 式 系统 。 衡 量 算法 优 劣 的 因素 包括 响应 时 间 、` 周 转 时 间 、 吞 吐 量 等 。 
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多 道 程序 设计 技术 大 大 提高 了 系统 的 资源 利用 率 , 但 是 使 多 个 程序 能 正确 地 并 发 执行 
是 相当 困难 的 。 并 发 是 所 有 问题 的 基础 ,也 是 操作 系统 设计 的 基础 。 并 发 包括 很 多 设计 问 
题 ,其 中 有 进程 间 通信 ,资源 共享 与 竞争 (如 内 存 、 文 件 .1/O 访问 )、 多 个 进程 活动 的 同步 以 
及 分 配给 进程 的 处 理 器 时 间 等 。 在 本 章 我 们 将 会 看 到 这 些 问 题 不 仅 会 出 现在 多 处 理 器 环境 
和 分 布 式 处 理 器 环境 中 ,也 会 出 现在 单 处 理 器 的 多 道 程序 设计 系统 中 。 


3.0 问题 导入 
当 多 个 进程 并 发 执行 时 ,可 能 会 同时 访问 一 些 共享 资源 ,此 时 如 何 保证 对 共享 资源 的 访 
问 是 正确 有 效 的 呢 ? 例如 ,有 多 个 售票 点 同时 出 售 飞机 票 , 如 何 设计 一 个 正确 的 并 发 程序 ， 
使 得 同一 航班 的 同一 张 票 不 被 出 售 给 多 人 呢 ? 


3.1 并 发 概述 


3.1.1 并 发 的 概念 


我 们 把 系统 中 可 并 发 执行 的 进程 称 为 “并 发 进程 ”, 并 发 进程 相互 之 间 可 能 是 无 关 的 ,也 
可 能 是 有 联系 的 。 如 果 一 个 进程 的 执行 不 影响 其 他 进程 的 执行 , 且 与 其 他 进程 的 进展 情况 
不 相关 , 即 它们 是 各 自 独立 的 , 则 说 这 些 并 发 进程 相互 之 间 是 无 关 的 。 显 然 ,无 关 的 并 发 进 
程 一 定 没 有 共享 的 变量 ,它们 分 别 在 各 自 的 数据 集合 上 操作 。 例 如 ,为 两 个 不 同 源 程序 进行 
编译 的 两 个 进程 可 以 是 并 发 执行 的 ,但 它们 之 间 却 是 无 关 的 。 因 为 这 两 个 进程 分 别 在 不 同 
的 数据 集合 上 为 不 同 的 源 程序 进行 编译 ,虽然 这 两 个 进程 可 交叉 地 占用 处 理 器 为 各 自 的 源 
程序 进行 编译 ,但 是 ,任何 一 个 进程 都 不 依赖 另 一 个 进程 。 甚 至 当 一 个 进程 发 现 被 编译 的 源 
程序 有 错误 时 ,也 不 会 影响 另 一 个 进程 继续 对 自己 的 源 程序 进行 编译 ,它们 是 各 自 独立 的 。 

然而 ,如 果 一 个 进程 的 执行 依赖 其 他 进程 的 进展 情况 ,或 者 说 一 个 进程 的 执行 可 能 影响 
其 他 进程 的 执行 结果 , 则 说 这 些 并 发 进程 相互 之 间 是 有 交往 的 .是 有 关 的 。 例 如 ,有 三 个 进 
程 , 即 读 进程 .处 理 进程 和 打印 进程 。 其 中 , 读 进程 每 次 启动 外 围 设备 读 一 批 信 息 并 把 读 入 
的 信息 存放 到 缓冲 区 ,处 理 进 程 对 存放 在 缓冲 区 中 的 信息 加 工 处 理 , 打 印 进程 把 加 工 处 理 后 
的 信息 打印 输出 。 这 三 个 进程 中 的 每 一 个 进程 的 执行 都 依赖 另 一 个 进程 的 进展 情况 : 只 有 
当 读 进程 把 一 批 信息 读 入 并 存 入 缓冲 区 后 ,处 理 进程 才能 对 它 进行 加 工 处 理 , 而 打印 进程 要 
等 信息 加 工 处 理 好 后 才能 把 它 输 出 ; 也 只 有 当 缓 冲 区 中 的 信息 被 打印 进程 取 走 后 , 读 进 程 
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才能 把 读 入 的 第 二 批 信息 再 存 入 缓冲 区 供 加 工 处 理 ; 如 此 循环 ,直至 所 有 的 信息 都 被 处 理 
且 打 印 输出 。 可 见 , 这 三 个 进程 相互 依赖 .相互 合作 ,它们 是 一 组 有 联系 的 并 发 进程 。 有 联 
系 的 并 发 进程 一 定 共享 某 些 资源 。 在 上 例 中 从 外 围 设备 上 读 入 的 信息 、 经 加 工 处 理 后 的 信 
息 、 存 放 信 息 的 缓冲 区 等 都 是 这 组 并 发 进程 的 共享 资源 。 


3.1.2 ”时序 错误 


一 个 进程 运行 时 由 于 自身 或 外 界 的 原因 而 可 能 被 中 断 , 且 断 点 是 不 固定 的 。 一 个 进程 
被 中 断后 ,哪个 进程 可 以 运行 ,被 中 断 的 进程 什么 时 候 再 去 占用 处 理 器 ,这 都 是 与 进程 调度 
算法 有 关 的 。 所 以 ,进程 执行 的 速度 不 能 由 自己 来 控制 ,对 于 有 联系 的 并 发 进程 来 说 ,可 能 
有 若干 并 发 进程 同时 使 用 共享 资源 , 即 一 个 进程 一 次 使 用 未 结束 另 一 个 进程 就 已 开始 使 用 ， 
形成 交 蔡 使 用 共享 资源 的 现象 。 如 果 对 这 种 情况 不 加 控制 ,就 可 能 出 现 与 时 间 有 关 的 错误 ， 
在 共享 资源 (变量 ) 时 就 会 出 错 , 并 得 到 不 正确 的 结果 。 请 观察 下 面 的 例子 。 

例 3-1 飞机 售票 问题 。 

假设 一 个 飞机 订 票 系统 有 两 个 终端 ,分 别 运行 进程 Tl 和 T2。 该 系统 公共 数据 区 中 的 
一 些 单元 Bj(j 二 1,2,…) 分 别 存 放 某 日 某 次 航班 机 票 的 余数 ,而 dl 和 d2 表示 进程 T1 和 
T2 执行 时 所 用 的 工作 单元 。 飞 机 售票 程序 如 下 。 


procedure Ti (i = 1, 2) 
di : integer ; 
begin 
[根据 旅客 订 票 要 求 找到 Bj] ; 
di := Bj; 
if di> = 1 then begin 
di:=di-1; 
Bj := di; 
[打印 一 张 票 ] 
end; 
else [提示 信息 " 票 已 售 完 ” 
end 
由 于 Tl 和 T2 是 两 个 可 同时 执行 的 并 发 进程 ,它们 在 同一 个 计算 机 系统 中 运行 ,共享 
同一 批 票 源 数据 ,因此 ,可 能 出 现 如 下 所 示 的 运行 情况 ( 设 Bj = m)。 
Tl:dl := Bj; 即 d =m(m>0) 
= Bj; 即 dz = m 
T2:d2 := d2 -1; 
= d2 ; [打印 一 张 票 ] ; 即 Bj = m-1 
怠 汪 粳 一 1 
Bj := dl ; [打印 一 张 票 ] ; 即 Bj= m-1 
显然 ,此 时 出 现 了 把 同一 张 票 卖 给 了 两 个 旅客 的 情况 ,两 个 旅客 都 买 到 一 张 同 天 同 次 航 
班 的 机 票 ,可 是 ,Bj 的 值 实际 上 只 减 去 了 1 ,造成 余 票数 的 不 正确 。 特 别 是 当 某 次 航班 只 有 
一 张 余 票 时 ,就 可 能 把 这 一 张 票 同时 售 给 两 位 旅客 ,这 是 不 允许 的 。 
例 3-2 主 存 管 理 问 题 。 


假定 有 两 个 并 发 进程 Borrow 和 Return 分 别 负 责 申 请 和 归还 主 存 


北 


mr 


中 ,x 表示 现 有 空闲 主 存 总 量 ,B 表示 申请 或 归还 的 主 存量 。 并 发 进程 算法 描 


X : integer; 
x := 1000; 
cobegin 
procedure Borrow ( B : integer) 
begin 
if B> x then [进程 进入 等 待 队 列 ,等 待 主 存 资源 ]; 
一 
[修改 主 存 分 配 表 , 进程 获得 主 存 资源 ]; 
end; 
procedure Return ( B : integer) 


begin 
下 By 
[修改 主 存 分 配 表 ]; 
[释放 等 待 主 存 资源 的 进程 ]; 
end; 


coend; 


由 于 Borrow 和 Return 共享 了 表示 主 存 物理 资源 的 临界 变量 x, 若 对 


述 如 下 。 


发 执行 不 加 限 


制 会 导致 错误 。 例 如 ,一 个 进程 调用 Borrow 申请 主 存 , 在 执行 了 比较 B 和 x 的 指令 后 ,发 
现 B 二 x, 但 在 执行 “进程 进入 等 待 队 列 ,等 待 主 存 资源 "前 另 一 个 进程 调用 Return 抢先 执 
行 ,归还 了 所 借 全 部 主 存 资源 。 这 时 ,由 于 申请 进程 还 未 成 为 等 待 状态 ,Return 中 的 “释放 
等 待 主 存 资源 的 进程 ”相当 于 空 操作 。 以 后 当 调 用 Borrow 的 进程 被 置 成 “等 待 主 存 资源 ” 
时 ,可 能 已 经 没有 其 他 进程 来 归还 主 存 资 源 了 ,从 而 ,申请 资源 的 进程 处 于 永远 等 待 状态 。 


例 3-3 自动 计算 问题 。 


某 交通 路 口 设 置 了 一 个 自动 计数 系统 ,该 系统 由 观察 者 (Observer) 进程 和 报告 者 


(Reporter) 进 程 组 成 。 观 察 者 进程 能 识别 汽车 ,并 对 通过 的 汽车 计数 ,报告 者 进程 定时 (可 


设 为 每 阳 一 小 时 ) 将 观察 者 的 计数 值 打 印 输出 ,每 次 打印 后 把 计数 值 清 *0”。 
发 执行 可 完成 对 每 小 时 汽车 流量 的 统计 ,这 两 个 进程 的 算法 描述 如 下 。 


count : integer; 
count := 0; 
cobegin 
procedure Observer 
begin 
L1 : [ observea car ]; 
count := count + 1; 
goto Ll; 
end; 
procedure Reporter 
begin 
print count; 
count := 0; 
end; 


coend; 


进程 Observer 和 Reporter 并 发 执行 时 可 能 有 如 下 两 种 情况 。 


两 个 进程 的 并 
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好 w 避 


握 作 系统 原理 


(1) 报告 者 进程 执行 时 无 汽车 通过 。 在 这 种 情况 下 ,报告 者 进程 把 上 一 小 时 通过 的 汽 
车 数 打印 输出 后 将 计数 器 清 “0”, 完 成 了 一 次 自己 承担 的 任务 。 此 后 ,有 汽车 通过 时 ,观察 者 
进程 重新 开始 对 一 个 新 时 间 段 内 的 流量 进行 统计 。 在 两 个 进程 的 配合 下 ,能 正确 统计 出 每 
小 时 通过 的 汽车 数量 。 

(2) 报告 者 进程 执行 时 有 汽车 通过 。 当 准点 时 ,报告 者 进程 工作 , 它 启动 了 打印 机 ,在 
等 待 打印 机 打印 输出 时 恰好 有 一 辆 卡车 通过 ,这 时 ,观察 者 进程 占用 处 理 器 ,把 计数 器 
count 的 值 又 增加 了 “1”。 之 后 ,报告 者 进程 在 打印 输出 后 继续 执行 count: 二 0。 于 是 ,报告 
者 进程 在 把 已 打印 的 count 值 清 *0? 时 ,同时 把 观察 者 进程 在 count 上 新 增加 的 “1? 也 清除 
了 。 如 果 在 报告 者 打印 期 间 连续 有 车 辆 通过 ,虽然 观察 者 都 把 它们 记录 到 计数 器 中 ,但 都 因 
报告 者 执行 count := 0 而 把 计数 值 丢失 了 ,使 统计 结果 严重 失实 。 

从 以 上 例子 可 以 看 出 ,由 于 并 发 进程 执行 的 随机 性 ,一 个 进程 对 另 一 个 进程 的 影响 是 不 
可 预测 的 。 由 于 它们 共享 了 资源 (变量 ), 当 在 不 同时 刻 交 蔡 访 问 资源 (变量 ) 时 就 可 能 造成 
结果 的 不 正确 。 造 成 不 正确 的 因素 与 进程 占用 处 理 器 的 时 间 、 执 行 的 速度 以 及 外 界 的 影响 
有 关 。 这 些 因素 都 与 时 间 有 关 , 所 以 把 它们 统称 为 “时序 错误 ”。 


3.1.3 临界 区 


有 联系 的 并 发 进程 执行 时 出 现 与 时 间 有 关 的 错误 ,其 根本 原因 是 对 共享 资源 (变量 ) 的 
使 用 不 加 限制 , 当 进 程 交叉 使 用 了 共享 资源 (变量 ) 时 就 可 能 造成 错误 。 为 了 使 并 发 进程 能 
正确 地 执行 ,必须 对 共享 变量 的 使 用 加 以 限制 。 

我 们 把 并 发 进程 中 与 共享 变量 有 关 的 程序 段 称 为 “临界 区 ”, 共 享 变量 所 代表 的 资源 称 
为 “临界 资源 ”, 多 个 并 发 进程 中 涉及 相同 共享 变量 的 那些 程序 段 称 为 “相关 临界 区 ”。 例 如 ， 
在 飞机 售票 系统 中 ,进程 T1 的 临界 区 为 : 


dl := Bj; 
if dl > = 1 then begin 
d :=dl-l 
Bj := dl; 
[打印 一 张 票 
end; 


进程 T2 的 临界 区 为 : 


d2 := Bj; 
if d2 > = 1 then begin 
dz := d2 -1; 
Bj := d2; 
[打印 一 张 票 ]; 
end; 
else [提示 信息 " 票 已 售 完 "]; 


这 两 个 临界 区 都 要 使 用 共享 变量 Bj, 故 属于 相关 临界 
而 在 自动 计数 系统 中 ,观察 者 进程 的 临界 区 是 : 


[al 


Count := count + 1; 


报告 者 进程 的 临界 


区 
和 


print count; 

count := 0; 

这 两 个 临界 区 都 要 使 用 共享 变量 count, 也 属于 相关 区 。 

如 果 有 进程 在 相关 临界 区 执行 时 ,不 让 另 一 个 进程 进入 相关 的 临界 区 执行 ,就 不 会 形成 
多 个 进程 对 相同 共享 变量 的 交叉 访问 ,于 是 就 可 避免 出 现 与 时 间 有 关 的 错误 。 例 如 ,观察 者 
和 报告 者 并 发 执行 时 , 当 报告 者 启动 打印 机 后 ,在 执行 count :二 0 之 前 ,虽然 观察 者 发 现 有 
汽车 通过 ,应 该 限制 它 进入 相关 临界 区 ( 即 暂 不 执行 count :一 count 十 1) ,直到 报告 者 执行 
了 count :一 0 退出 临界 区 。 当 报告 者 退出 临界 区 后 ,观察 者 再 进入 临界 区 执行 ,这 样 就 不 
会 交 蔡 地 修改 count 值 ,而 观察 到 的 汽车 数 被 统计 在 下 一 个 时 间 段 内 ,也 不 会 出 现 数据 丢 
失 。 可 见 , 只 要 对 涉及 共享 变量 的 临界 区 互 斥 执行 ,就 不 会 出 现 与 时 间 有 关 的 错误 。 因 而 ， 
对 若干 进程 共享 某 一 资源 (变量 ) 的 相关 临界 区 的 管理 应 满足 如 下 三 个 要 求 。 

(1) 一 次 最 多 让 一 个 进程 在 临界 区 执行 , 当 有 进程 在 临界 区 执行 时 ,其 他 想 进入 临界 区 
执行 的 进程 必须 等 待 。 

(2) 任何 一 个 进入 临界 区 执行 的 进程 必须 在 有 限 的 时 间 内 退出 临界 区 , 即 任何 一 个 进 
程 都 不 应 该 无 限 地 逗留 在 自己 的 临界 区 中 。 

(3) 不 能 强迫 一 个 进程 无 限 地 等 待 进入 它 的 临界 区 , 即 有 进程 退出 临界 区 时 应 让 一 个 
等 待 进入 临界 区 的 进程 进入 它 的 临界 区 。 


3.1.4 进程 的 互 斥 


进程 的 互 斥 是 指 当 有 若干 进程 都 要 使 用 某 一 共享 资源 时 ,任何 时 刻 最 多 只 允许 一 个 进 
程 去 使 用 ,其 他 要 使 用 该 资源 的 进程 必须 等 待 ,直到 占用 资源 者 释放 该 资源 。 

实际 上 ,共享 资源 的 互 斥 使 用 就 是 限定 并 发 进程 互 斥 地 进入 相关 临界 区 。 如 果 能 提供 
一 种 方法 来 实现 对 相关 临界 区 的 管理 , 则 就 可 实现 进程 的 互 斥 。 实 现 对 相关 临界 区 管理 的 
方法 有 多 种 ,如 可 采用 标志 方式 .上 锁 开锁 方式 .PV 操作 方式 和 管 程 方式 等 。 

在 这 里 , 先 介绍 几 种 硬件 实现 互 斥 的 方案 。 在 这 些 方案 中 , 当 一 个 进程 在 临界 区 中 更 新 
共享 资源 时 ,其 他 进程 将 不 会 进入 其 临界 区 ,从 而 保证 程序 的 正确 执行 。 

1. 中 断 禁 用 

在 单 处 理 器 机 器 中 ,并 发 进程 不 能 重合 执行 ,只 能 交替 执行 。 此 外 ,一 个 进程 将 一 直 运 
行 ,直到 它 调用 了 一 个 系统 服务 或 被 中 断 。 因 此 为 保证 互 斥 ,只 需要 保证 一 个 进程 不 被 中 断 
就 可 以 了 ,这 种 能 力 可 以 通过 系统 内 核 为 启用 和 禁用 中 断定 义 的 原 语 来 提供 。 一 个 进程 可 
以 通过 下 面 的 方法 实施 互 斥 。 


Xx| 


while (true) { 
/* 禁用 中 断 */; 
/x 临界 区 * /5 
/x* 启用 中 断 */; 
/x 其 余部 分 */; 
} 


由 于 临界 区 不 能 被 中 断 , 故 可 以 保证 互 斥 , 但 是 ,该 方法 的 代价 非常 高 ,由 于 处 理 器 被 限 
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制 于 只 能 交 蔡 执行 程序 ,因此 执行 的 效率 将 会 有 明显 的 降低 。 另 一 个 问题 是 该 方法 不 能 
于 多 处 理 器 结构 中 , 当 一 个 计算 机 系统 包括 多 个 处 理 器 时 ,就 有 可 能 有 一 个 以 上 的 进程 同时 
执行 ,在 这 种 情况 下 ,禁用 中 断 是 不 能 保证 互 斥 的 。 

2. 其 他 处 理 方法 

在 多 处 理 器 配置 中 , 几 个 处 理 器 共享 内 存 。 在 这 种 情况 下 ,不 存在 主 / 从 关系 ,处 理 器 间 
的 行为 是 无 关 的 ,表现 出 一 种 对 等 关系 ,处 理 器 之 间 没 有 支持 互 斥 的 中 断 机 制 。 

在 硬件 级 别 上 ,对 存储 单元 的 访问 排斥 对 相同 单元 的 其 他 访问 。 基 于 这 一 点 ,处 理 器 的 
设计 者 提出 了 一 些 机 器 指令 ,用 于 保证 两 个 动作 的 原子 性 ,如 在 一 个 取 指 令 周 期 中 对 一 个 存 
储 器 单元 的 读 和 写 是 否 唯一 。 在 该 指令 执行 的 过 程 中 ,任何 其 他 指令 访问 内 存 将 被 阻止 ,而 
且 这 些 动作 在 一 个 指令 周期 中 完成 。 

本 节 给 出 了 两 种 最 常见 的 指令 : 比较 和 交换 指令 ,exchange 指令 。 

1) 比较 和 交换 指令 

比较 和 交换 指令 定义 如 下 。 


int compare_and_swap (int * word，int testval, int newval) 
{ 

int oldval; 

oldval = xword 

if (oldval == testval) * word = newval; 

return oldval; 


; 


该 指令 的 一 个 版 本 是 用 一 个 测试 值 (testval) 检 查 一 个 内 存单 元 ( x* word)。 如 果 该 内 
存单 元 的 当前 值 是 testval, 就 用 newval 取代 该 值 ; 否则 保持 不 变 。 该 指令 总 是 返回 旧 内 存 
值 ,因此 ,如 果 返 回 值 与 测试 值 相同 , 则 表示 该 内 存单 元 已 被 更 新 。 由 此 可 见 这 个 原子 指令 
两 部 分 组 成 : 比较 内 存单 元 值 和 测试 值 ; 如 果 值 相同 , 则 产生 交换 (Swap) ,整个 比较 和 交 
换 功 能 按 原子 操作 执行 , 即 它 不 接受 中 断 。 

该 指令 的 另 一 个 版 本 返回 一 个 布尔 (Boolean) 值 : 交换 发 生 时 为 真 (True); 否则 为 假 
(False)。 几 乎 所 有 处 理 器 家 族 (x86、IA64、SPARC 和 IBMZ 系列 机 等 ) 中 都 支持 该 指令 的 
某 个 版 本 ,而 且 多 数 操作 系统 都 利用 该 指令 支持 并 发 。 
图 3-1(a) 给 出 了 基于 使 用 这 个 指令 的 互 斥 规程 。 共 享 变量 bolt 被 初始 化 为 0。 唯一 可 
以 进入 临界 区 的 进程 是 发 现 bolt 等 于 0 的 那个 进程 。 所 以 有 试图 进入 临界 区 的 其 他 进程 
进入 忙 等 待 模式 。 术 语 忙 等 待 (Busy Waiting) 或 自 旋 等 待 (Spin Waiting) 指 的 是 这 样 一 种 
技术 : 进程 在 得 到 临界 区 访问 权 之 前 , 它 只 能 继续 执行 测试 变量 的 指令 来 得 到 访问 权 , 除 此 
之 外 不 能 做 其 他 事情 。 当 一 个 进程 离开 临界 区 时 , 它 把 bolt 重 置 为 0, 此 时 只 有 一 个 等 待 进 
程 被 允许 进入 临界 区 。 进 程 的 选择 取决 于 哪个 进程 正好 执行 紧 接着 的 比较 和 交换 指令 。 
2) exchange 指令 
exchange 指令 定义 如 下 。 


void exchange (int * register, int x memory) 
{ 
int temp; 


temp = * memory; 


x memory = *register; 
< register = temp; 
} 
该 指令 交换 一 个 寄存 器 的 内 容 和 一 个 存储 单元 的 内 容 。Intel IA-32(Pentium) 和 IA-64 
(Itanium) 体 系 结构 都 含有 XCHG 指令 。 


* Program mutualexclusion */ /* program mutualexclusion */ 
const int n=/#* 进程 个 数 */; int constn =/# ”进程 个 数 */; 
int bolt; int bolt; 
voidP (int)) void P(int i) 
{ { 
while (true) { int keyi= 1; 
while (compare_and_swap(bolt, 0, 1) 一 1) while (true) { 
记 不 做 任何 事 */; do exchange (&keyi, &bolt) 
卡 临 界 区 */; while (keyi {= 0); 
bolt= 0; 话 临 界 区 */; 
启 其 余部 分 */; bolt=0; 
} 上 其 余部 分 4/; 
b } 
void main() } 
{ void main() 
bolt=0; { 
parbegi n (P(1),P(2),...,P((n)); bolt=0 


parbegin (P(1),P(2),...,PC(n)); 
} } 
(a) 比较 和 交换 指令 (b) 交换 指令 


图 3-1 对 互 斥 的 硬件 支持 


图 3-1(b) 显 示 了 基于 exchange 指令 的 互 斥 规程 : 共享 变量 bolt 被 初始 化 为 0, 每 个 进 
程 都 使 用 一 个 局 部 变量 key 且 初始 化 为 1。 唯一 可 以 进入 临界 区 的 进程 是 发 现 bolt 等 于 0 
的 那个 进程 。 它 通过 把 bolt 置 为 1 排斥 所 有 其 他 进程 进入 临界 区 。 当 一 个 进程 离开 临界 
区 时 , 它 把 bolt 重 置 为 0, 允 许 另 一 个 进程 进入 它 的 临界 区 。 

注意 ,由 于 变量 初始 化 的 方式 及 exchange 算法 的 本 质 ,下 面 的 表达 式 总 是 成 立 的 。 

bolt 十 > )keyi =n 


如 果 bolt = 0, 则 没有 任何 一 个 进程 在 它 的 临界 区 中 ; 如 果 bolt = 1, 则 只 有 一 个 进程 
在 临界 区 中 , 即 key 的 值 等 于 0 的 那个 进程 。 

使 用 专门 的 机 器 指令 实施 互 斥 具有 以 下 优点 。 

(1) 适用 于 在 单 处理 器 或 共享 内 存 的 多 处 理 器 上 的 任何 数目 的 进程 。 

(2) 非常 简单 且 易 于 证 明 。 

(3) 可 用 于 支持 多 个 临界 区 ,每 个 临界 区 可 以 用 它 自己 的 变量 定义 。 

但 是 , 它 也 有 一 些 严重 的 缺点 。 

(1) 使 用 了 忙 等 待 : 当 一 个 进程 正在 等 待 进入 临界 区 时 , 它 会 继续 消耗 处 理 器 时 间 。 

(2) 可 能 饥饿 : 当 一 个 进程 离开 一 个 临界 区 并 且 有 多 个 进程 正在 等 待 时 ,选择 哪 一 个 
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等 待 进程 是 任意 的 ,因此 某 些 进程 可 能 被 无 限 地 拒绝 进入 。 

(3) 可 能 死 锁 : 考虑 单 处 理 器 中 的 下 列 情况 。 进 程 Pl 执行 专门 指令 (如 compare & 
swap、exchange) 并 进入 临界 区 ,然后 Pl 被 中 断 并 把 处 理 器 让 给 具有 更 高 优先 级 的 P2。 如 
果 P2 试图 使 用 同一 资源 ,由 于 互 斥 机 制 , 它 将 被 拒绝 访问 。 因 此 , 它 会 进入 忙 等 待 循环 。 
但 是 ,由 于 Pl 比 P2 的 优先 级 低 , 它 将 永远 不 会 被 调度 执行 。 


3.2 PV 操作 


上 面 所 介绍 的 硬件 实现 互 斥 属于 忙 等 待 的 互 斥 ,本 节 介 绍 怎样 用 PV 操作 来 管理 相关 
临界 区 , 亦 即 用 PV 操作 实现 进程 的 互 斥 。 


3.2.1 信号 量 与 PV 操作 


1. 信号 量 

信号 量 的 概念 和 PV 操作 是 荷兰 科学 家 E. W. Dijkstra 提出 来 的 。 信 号 是 交通 管理 中 
的 一 种 常用 设备 ,交通 管理 人 员 利 用 信号 颜色 的 变化 来 实现 交通 管理 。 在 操作 系统 中 ,信号 
量 S 是 一 个 整数 。 当 S 大 于 等 于 零 时 ,代表 可 供 并 发 进程 使 用 的 资源 实体 数 ; 当 S 小 于 零 
时 , 则 |S| 表 示 正 在 等 待 使 用 资源 实体 的 进程 数 。 建 立 一 个 信号 量 必须 说 明 此 信号 量 所 代 
表 的 意义 并 且 赋 初 值 。 除 赋 初 值 外 ,信号 量 仅 能 通过 PV 操作 来 访问 。 

信号 量 按 其 用 途 可 分 为 以 下 两 种 。 

(1) 公用 信号 量 , 联 系 一 组 并 发 进程 ,相关 的 进程 均 可 在 此 信号 量 上 进行 P 操作 和 V 
操作 , 初 值 常 常 为 1, 用 于 实现 进程 互 斥 ,也 称 为 互 斥 信号 量 。 

(2) 私有 信号 量 , 联 系 一 组 并 发 进程 , 仅 允许 拥有 此 信号 量 的 进程 执行 P 操作 ,而 其 他 
相关 进程 可 在 其 上 施行 V 操作 。 初 值 常常 为 0 或 正 整数 ,多 用 于 实现 进程 同步 ,也 称 为 资 
源 信号 量 。 

2. PV 操作 

PV 操作 是 由 两 个 操作 , 即 P 操作 和 V 操作 组 成 。P 操作 和 V 操作 是 两 个 在 信号 量 上 
进行 操作 的 过 程 ,假定 用 S 表示 信号 量 . 则 把 这 两 个 过 程 记 作 P(S) 和 V(S) ,它们 的 定义 如 下 。 


procedure P(S: Semaphore) 
begin 
S:=S— 1; 
if S<0 then Wait(S); 
end; 
procedure V(S:Semaphore) 
begin 
Si:= 8S+l 
if S<0 then Release(S); 
end; 


其 中 ,Wait(S) 表 示 将 调用 P(S) 过 程 的 进程 置 成 “等 待 信号 量 S” 的 状态 , 且 将 其 排 入 等 
待 队 列 ; Release(S) 表 示 释 放 一 个 “等 待 信号 量 S” 的 进程 ,使 该 进程 从 等 待 队 列 退 出 并 加 入 
就 绪 队 列 中 。 


要 用 PV 操作 来 管理 共享 资源 ,首先 要 确保 PV 操作 自身 执行 的 正确 性 。 由 于 P(S) 和 
V(S) 都 是 在 同一 个 信号 量 S 上 操作 ,为 了 使 得 它们 在 执行 时 不 发 生 交叉 访问 信号 量 S 而 可 
能 出 现 的 错误 ,约定 PCS) 和 V(S) 必 须 是 两 个 不 可 被 中 断 的 过 程 , 即 让 它们 在 屏蔽 中 断 下 执 
行 。 我 们 把 不 可 被 中 断 的 过 程 称 为 " 原 语 ”, 于 是 了 P 操作 和 V 操作 实际 上 是 “P 操作 原 语 > 和 
“V 操作 原 语 ”。 在 有 的 教材 上 ,P、V 操作 也 分 别称 为 Wait() 和 Signal() 操 作 。 

P 操作 的 主要 动作 如 下 。 

(1) S 减 1; 

(2) 车 S 减 1 后 仍 大 于 或 等 于 零 , 则 进程 继续 执行 ; 

(3) 若 S 减 1 后 小 于 零 , 则 该 进程 被 阻塞 后 放 入 等 待 该 信号 量 的 等 待 队 列 中 ,然后 转 进 
程 调度 ,如 图 3-2 所 示 。 

V 操作 的 主要 动作 如 下 。 

I 玉 

(2) 若 结果 大 于 零 , 则 进程 继续 执行 ; 

(3) 若 结果 小 于 或 等 于 零 , 则 从 该 信号 的 等 待 队列 中 释放 一 个 等 待 进程 ,然后 再 返回 原 
进程 继续 执行 或 转 进 程 调度 ,如 图 3-3 所 示 。 


| 调用 进程 入 等 等 队列 


转 进程 调度 转 进程 调度 


图 3-2 P 操 作 功能 图 3-3 V 操作 功能 


个 进程 
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S 的 初 值 可 定义 为 0、1 或 其 他 整数 ,在 系统 初始 化 时 确定 。 从 信号 量 和 PV 操作 的 定 
义 可 以 获得 如 下 推论 。 

推论 1: 若 信 号 量 S 为 正 值 , 则 该 值 等 于 S 所 代表 的 实际 可 以 使 用 的 物理 资源 数 。 

推论 2: 若 信 号 量 S 为 负 值 , 则 其 绝对 值 等 于 对 信号 量 S 实施 P 操作 而 被 阻塞 并 进入 
信号 量 S 等 待 队 列 的 进程 数 。 

推论 3: 通常 P 操 作 意味 着 请 求 一 个 资源 ,V 操作 意味 着 释放 一 个 资源 。 在 一 定 条 件 
下 ,P 操作 代表 阻塞 进程 操作 ,而 V 操作 代表 唤醒 被 阻塞 进程 的 操作 。 


3.2.2 用 PV 操作 实现 进程 互 斥 
用 PV 操作 可 实现 并 发 进程 的 互 斥 ,其 步骤 如 下 。 
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(1) 设立 一 个 互 斥 信号 量 S 表示 临界 区 ,其 取 值 范围 为 1,0, 一 1,…, 其 中 ,S 一 1 表示 
无 并 发 进程 进入 S 临界 区 ; S=0 表示 已 有 一 个 并 发 进程 进入 S 临界 区 ; S 等 于 负数 表示 已 
有 一 个 并 发 进程 进入 S 临界 区 , 且 有 |S| 个 进程 等 待 进入 S 临界 区 。S 的 初 值 为 1。 

(2) 用 PV 操作 表示 对 S 临界 区 的 申请 和 释放 ,在 进入 临界 区 之 前 ,通过 了 操作 进行 申 
请 ,在 退出 临界 区 之 后 ,通过 V 操作 释放 。 


A 进程 B 进程 
5 (BY P(S); 
临界 区 ; 临界 区 ; 
V(s); V(S); 
下 面 请 看 几 个 实例 。 


例 3-4 用 PV 操作 管理 飞机 售票 问题 。 


S: Semaphore; 


S:= 1; 
cobegin 
procedure Ti (i = 1, 2) 
di : integer; 
begin 
[根据 旅客 订 票 要 求 找到 Bj]; 
P(S); 
di := Bj 
if di > = 1 then begin 
di := di-1; 
Bj := di; 
V(S); 
[输出 一 张 票 ]; 
end; 
else begin 
Vv(S); 
提示 信息 " 票 已 售 完 "]; 
end; 
end; 


coend; 


例 3-5 PV 操作 管理 主 存 问题 。 


x: integer; 
S: Semaphore; 
* = 1000; 8 := 1; 
cobegin 
procedure Borrow (B: integer) 


begin 
P(S); 
if B> x then [进程 进入 等 待 队 列 ,等 待 主 存 资源 ]; 
x:= x— B; 


[修改 主 存 分 配 表 , 进程 获得 主 存 资源 ]; 


V(S); 
end; 
procedure Return (B: integer) 

begin 
P(S); 
六 
[修改 主 存 分 配 表 ]; 
V(S)， 
[释放 等 待 主 存 资源 的 进程 ]; 


coend; 


例 3-6 PV 操作 管理 自动 计数 问题 。 


count: integer; 
S: Semaphore; 
count := 0;S:= 1; 
cobegin 
Pprocedure Observer 
begin 
L1: [observer a car]; 
Pp(S); 
count := count + 1; 
Vv(s); 
goto Ll1; 
end; 
procedure Reporter 
begin 
P(S); 
print count; 
count := 0; 
Vv(S); 
end; 
coend; 


例 3-7 用 PV 操作 解决 5 个 哲学 家 吃 通 心 面 问题 。 


有 5 个 哲学 家 围 坐 在 一 张 圆桌 旁 , 桌 子 中 央 有 一 盘 通 心 面 , 每 人 面前 有 一 只 空 盘子 ,每 
两 人 之 间 放 一 根 饶 子 。 每 个 哲学 家 思考 .饥饿 ,然后 欲 吃 通 心 面 。 为 了 吃 面 , 每 个 哲学 家 必 


须 获得 两 根 筷子 , 且 每 人 只 能 直接 从 自己 左边 或 右边 取得 
筷子 (如 图 3-4 所 示 ) 。 

在 这 道 经 典 题 目 中 ,每 一 根 筷子 都 是 必须 互 斥 使 用 
的 ,因此 ,应 为 每 根 筷子 设置 一 个 互 斥 信号 量 Si(i 一 0,1， 
2,3,4) , 初 值 均 为 1, 当 一 个 哲学 家 吃 通 心 面 之 前 必须 获 
得 自己 左边 和 右边 的 两 根 筷子 , 即 执行 两 个 P 操作; 吃 完 
通 心 面 后 必须 放下 筷子 , 即 执行 两 个 V 操作 。 


S0，S1，S2，S3，S4 : Semaphore; 
ys ly 
cobegin 


Ww 


O x 

gk 
SS 
图 3-4 5 个 哲学 家 进餐 问题 
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procedure PHi(i = 0,1,2,3,4) 

begin 

Ll: 
[思考 ]; 
P(Si); 
P((Si + 1) mod 5 ); 
[ 吃 通 心 面 ]; 
V(Si); 
V((Si + 1) mod 5 ); 
goto D1; 

end; 

coend; 


3.3 进程 同步 


3.3.1 同步 的 概念 


利用 信号 量 我 们 解决 了 进程 的 互 斥 问题 ,但 互 斥 主要 是 解决 并 发 进程 对 临界 区 的 使 用 
问题 。 这 种 基于 临界 区 控制 的 交互 作用 是 比较 简单 的 ,只 要 诸 进 程 对 临界 区 的 执行 时 间 互 
斥 , 每 个 进程 就 可 忽略 其 他 进程 的 存在 和 作用 。 此 外 ,还 需要 解决 异步 环境 下 的 进程 同步 问 
题 。 所 谓 异 步 环 境 , 是 指 相互 合作 的 一 组 并 发 进程 ,其 中 每 一 个 进程 都 以 各 自 独立 的 、 不 
可 预知 的 速度 向 前 推进 ,但 它们 又 需要 密切 合作 以 实现 一 个 共同 的 任务 , 即 彼此 “知道 
相互 的 存在 和 作用 。 例 如 ,为 了 把 原始 的 一 批 记录 加 工 成 当前 需要 的 记录 ,创建 了 两 个 
进程 , 即 进程 A 和 进程 B。 进 程 A 启动 输入 设备 不 断 地 读 记 录 , 每 读 出 一 个 记录 就 交 给 
进程 B 去 加 工 , 直 至 所 有 记录 都 处 理 结束 。 为 此 ,系统 设置 了 一 个 容量 为 存放 一 个 记录 
的 缓冲 器 ,进程 A 把 读 出 的 记录 存 入 缓冲 器 ,进程 B 从 缓冲 器 中 取出 记录 加 工 , 如 图 3-5 


所 示 。 
读 存 取 加 工 
“© "EO" 


进程 缓冲 器 进程 
图 3-5 进程 协作 


进程 A 和 进程 B 是 两 个 并 发 进程 ,它们 共享 缓冲 器 ,如 果 两 个 进程 不 相互 制约 就 会 造 
成 错误 。 当 进程 A 的 执行 速度 超过 进程 B 的 执行 速度 时 ,可 能 进程 A 把 一 个 记录 存 入 缓冲 
器 后 ,在 进程 B 还 没有 取 走 前 ,进程 A 又 把 新 读 出 的 一 个 记录 存 入 缓冲 器 ,后 一 个 记录 就 把 
前 一 个 尚未 取 走 的 记录 覆盖 了 .造成 记录 的 丢失 。 当 进程 B 的 执行 速度 超过 进程 A 的 执行 
速度 时 ,可 能 进程 B 从 缓冲 器 取出 一 个 记录 并 加 工 后 ,进程 A 还 没有 把 下 一 个 新 记录 存 入 
缓冲 器 ,而 进程 B 却 又 从 缓冲 器 中 去 取 记 录 , 造 成 重复 地 取 同 一 个 记录 加 工 。 

用 进程 互 斥 的 办 法 不 能 克服 上 述 两 种 错误 。 事 实 上 ,进程 A 和 进程 B 虽然 共享 缓冲 
器 ,但 它们 都 是 在 无 进程 使 用 缓冲 器 时 才 向 缓冲 器 存 记录 或 从 缓冲 器 取 记 录 的 。 也 就 是 说 ， 
它们 在 互 斥 使 用 共享 缓冲 器 的 情况 下 仍 会 发 生 错 误 , 引 起 错误 的 根本 原因 是 它们 之 间 的 相 
对 速度 。 可 以 采用 互通 消息 的 办 法 来 控制 执行 速度 ,使 相互 协作 的 进程 正确 工作 。 


两 个 进程 应 该 按照 如 下 原则 协作 。 

(1) 进程 A 把 一 个 记录 存 入 缓冲 区 后 ,应 向 进程 B 发 送 “ 缓 冲 器 中 有 等 待 处 理 的 记录 ” 
的 消息 。 

(2) 进程 B 从 缓冲 器 中 取出 记录 后 ,应 向 进程 A 发 送 “ 缓 冲 器 中 的 记录 已 取 走 ”的 
消息 。 

(3) 进程 A 只 有 在 得 到 进程 B 发 送 来 的 “缓冲 器 中 的 记录 已 取 走 ”的 消息 后 ,才能 把 下 
一 个 记录 再 存 入 缓冲 器 。 和 否则 进程 A 等 待 ,直到 消息 到 达 。 

(4) 进程 只 有 在 得 到 进程 A 发 送 来 的 “缓冲 器 中 有 等 待 处 理 ” 的 消息 后 ,才能 从 缓冲 
器 中 取出 记录 并 加 工 。 否 则 进程 B 等 待 ,直到 消息 到 达 。 

由 于 每 个 进程 都 是 在 得 到 对 方 的 消息 后 才 去 使 用 共享 的 缓冲 器 ,所 以 不 会 出 现 记录 的 
丢失 和 记录 的 重复 处 理 。 

因此 ,进程 的 同步 是 指导 并 发 进程 之 间 存 在 一 种 制约 关系 ,一 个 进程 的 执行 依赖 另 
一 个 进程 的 消息 , 当 一 个 进程 没有 得 到 另 一 个 进程 的 消息 时 应 等 待 ,直到 消息 到 达 时 才 
被 唤醒 。 


3.3.2 PV 操作 实现 进程 同步 


要 实现 进程 的 同步 就 必须 提供 一 种 机 制 , 该 机 制 能 把 其 他 进程 需要 的 消息 发 送出 去 ,也 
能 测试 自己 需要 的 消息 是 否 到 达 。 把 能 实现 进程 同步 的 机 制 称 为 同步 机 制 ,不 同 的 同步 机 
制 实现 同步 的 方法 也 不 同 ,PV 操作 和 管 程 是 两 种 典型 的 同步 机 制 。 本 节 介 绍 怎样 用 PV 操 
作 实 现 进程 间 的 同步 。 

我 们 已 经 知道 怎样 用 PV 操作 来 实现 进程 的 互 斥 。 事 实 上 ,PV 操作 不 仅 是 实现 进程 互 
斥 的 有 效 工具 ,而 且 还 是 一 个 简单 而 方便 的 同步 工具 。 用 一 个 信号 量 与 一 个 消息 联系 起 来 ， 
当 信号 量 的 值 为 "0" 时 表示 期 望 的 消息 尚未 产生 , 当 信号 量 的 值 为 非 “0" 时 表示 期 望 的 消息 
已 经 存在 。 假 定 用 信号 量 S 表示 某 个 消息 ,现在 来 看 看 怎样 用 PV 操作 达到 进程 同步 的 
目的 。 

1. 调用 了 操作 测试 消息 是 否 到 达 

任何 进程 调用 P 操作 可 测试 到 自己 所 期 望 的 消息 是 否 已 经 到 达 。 若 消息 尚未 产生 则 
S 一 0, 调 用 P(S) 后 ,P(S) 一 定 让 调用 者 成 为 等 待 信号 量 S 的 状态 , 即 调用 者 此 时 必定 等 待 
直到 消息 到 达 ; 若 消息 已 经 存在 则 S 取 0, 调用 P(S) 后 进程 不 会 成 为 等 待 状态 而 可 继续 执 
行 , 即 进程 测试 到 自己 期 望 的 消息 已 经 存在 。 

2. 调用 V 操作 发 送 消息 

任何 进程 要 向 其 他 进程 发 送 消息 时 可 调用 V 操作 。 若 调用 V 操作 之 前 ==0, 表 示 消 
息 尚未 产生 且 无 等 待 消息 的 进程 ,这 时 调用 V(S) 后 执行 S :一 S+1 使 S 冯 0, 即 意味 着 消息 
已 存在 ; 若 调用 V 操作 之 前 S 过 0, 表 示 消 息 未 产生 前 已 有 进程 在 等 待 消息 ,这 时 调用 
V(S) 后 将 释放 一 个 等 待 消息 者 , 即 表示 该 进程 等 待 的 消息 已 经 到 达 可 以 继续 执行 。 

在 用 PV 操作 实现 同步 时 ,一 定 要 根据 具体 的 问题 来 定义 信号 量 和 调用 了 操作 或 V 操 
作 。 一 个 信号 量 与 一 个 消息 联系 在 一 起 , 当 有 多 个 消息 时 必须 定义 多 个 信号 量 ; 测试 不 同 
的 消息 是 否 到 达 或 发 送 不 同 的 消息 时 ,应 对 不 同 的 信号 量 调用 P 操作 或 V 操作 。 
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3.3.3 生产 者 -消费 者 问题 


生产 者 -消费 者 问题 是 一 个 典型 的 同步 例子 。 假 定 有 一 个 生产 者 和 一 个 消费 者 ,他 们 共 
个 缓冲 器 ,生产 者 不 断 地 生产 物品 ,每 生产 一 件 物品 就 要 存 入 缓冲 器 ,但 缓冲 器 中 每 次 
只 能 存 入 一 件 物品 ,只 有 当 消 费 者 把 物品 取 走 后 ,生产 者 才能 把 下 一 件 物品 存 入 缓冲 器 。 同 
样 地 ,消费 者 要 不 断 地 从 缓冲 器 取出 物品 消费 , 当 缓 冲 器 中 有 物品 时 他 就 可 以 去 取 , 每 取 走 
一 件 物品 后 必须 等 生产 者 再 放 一 件 物品 后 才能 再 取 。 

在 这 个 问题 中 ,生产 者 要 向 消费 者 发 送 “ 缓 冲 器 中 有 物品 ”的 消息 ,而 消费 者 要 向 生产 者 
发 送 “ 可 把 物品 存 入 缓冲 器 ”的 消息 。 用 PV 操作 实现 生产 者 -消费 者 之 间 的 同步 ,应 该 定义 
两 个 信号 量 , 分 别 表示 两 个 消息 。 我 们 把 这 两 个 信号 量 定义 为 sPdt 和 sGet, 它 们 的 含义 
如 下 。 

(1) sPdt: 表示 是 否 可 以 把 物品 存 入 缓冲 器 ,由 于 缓冲 器 中 只 能 放 一 件 物品 ,系统 初始 
化 时 应 允许 放 入 物品 ,所 以 sPdt 的 初 值 应 为 "1”。 

(2) sGet: 表示 缓冲 器 中 是 否 存 有 物品 ,显然 ,系统 初始 化 时 缓冲 器 中 应 该 无 物品 ,所 
以 sGet 的 初 值 应 为 “0”。 

对 生产 者 来 说 ,生产 一 件 物品 后 应 调用 PCsPdt) , 当 缓 冲 器 中 允许 放 物 品 时 (sPdt 二 1), 则 
在 调用 PC(sPdt) 后 可 以 把 物品 存 入 缓冲 器 (此 时 sPdt 的 值 已 变 为 “0”)。 生 产 者 把 一 件 物品 
存 入 缓冲 器 后 ,又 可 继续 去 生产 物品 ,但 若 消 费 者 尚未 取 走 上 一 件 物品 (这 时 sPdt 维持 为 
“0”) ,而 生产 者 要 把 生产 的 物品 存 入 缓冲 器 时 调用 P(sPdt) 后 将 成 为 等 待 状态 ,阻止 它 把 物 
品 存 入 缓冲 器 。 生 产 者 在 缓冲 器 中 每 存 入 一 件 物品 后 ,应 调用 V(sGet) 把 缓冲 器 中 有 物品 
的 消息 告诉 消费 者 (调用 V(sGet) 后 ,sGet 的 值 从 “0” 变 为 *1”)。 
对 消费 者 来 说 , 取 物 品 前 应 查看 缓冲 器 中 是 否 有 物品 , 即 调用 P(sGet)。 若 缓冲 器 中 尚 
无 物品 (sGet 仍 为 "0”), 则 调用 PCsGet) 后 消费 者 等 待 ,不 能 去 取 物 品 ,直到 生产 者 存 入 一 件 
物品 后 发 送 有 物品 的 消息 时 才 唤 醒 消费 者 。 若 缓冲 器 中 已 有 物品 (sGet 为 “1”), 则 调用 
P(CsGet) 后 消费 者 可 继续 执行 ,从 缓冲 器 中 去 取 物 品 。 消 费 者 从 缓冲 器 中 每 取 走 一 件 物品 
后 应 调用 V(sPdt) ,通知 生产 者 缓冲 器 中 物品 已 取 走 ,可 以 存 入 一 件 新 物品 。 

例 3-8 生产 者 和 消费 者 并 发 执行 时 ,用 PV 操作 作为 同步 机 制 可 按 如 下 方式 管理 。 


sPdt, sGet :Semaphore; 
Buffer: integer; 
sPdt := 1; sGet := 0; 
cobegin 
procedure producer 
begin 
L1: [Produce a product]; 
P(sPdt); 
buffer := product; 
V(sGet); 
goto Ll1; 
end; 
procedure consumer 


begin 


L2: P(sGet); 
[Take a product from buffer]; 
V(sPdt); 
[Consume]; 
end; 

coend; 

请 注意 ,生产 者 生产 物品 的 操作 和 消费 者 消费 物品 的 操作 是 各 自 独立 的 ,只 是 在 访问 公 
用 的 缓冲 器 把 物品 存 入 或 取出 时 才 要 互通 消息 。 所 以 ,测试 消息 是 否 到 达 和 发 送 消息 的 P 
操作 与 V 操作 应 该 分 别 在 访问 共享 缓冲 器 之 前 和 之 后 。 

如 果 一 个 生产 者 和 一 个 消费 者 共享 的 缓冲 器 容量 为 可 以 存放 件 物品 (n 之 1) ,那么 
只 要 把 信号 量 sPdt 的 初 值 定 为 “n”,sGet 的 初 值 仍 为 “0”。 当 缓冲 器 中 没有 放 满 n 件 物品 
时 ,生产 者 调用 PCsPdt) 后 都 不 会 成 为 等 待 状态 而 可 以 把 生产 出 来 的 物品 存 入 缓冲 器 。 但 
当 缓 冲 器 中 已 经 及 件 物品 时 (sPdt 值 为 "0”) ,生产 者 再 想 存 入 一 件 物品 将 被 拒绝 。 生 产 
者 每 存 入 一 件 物品 后 ,由 于 调用 V(sGet) 发 送 消息 , 故 sGet 的 值 表示 缓冲 器 中 可 供 消费 的 
物品 数 。 只 要 sGet 夭 0, 消费 者 调用 P(sGet) 后 总 可 以 去 取 物 品 ,每 取 走 一 件 物品 后 调用 
V(sPdt) , 便 增加 了 一 个 可 以 用 来 存放 物品 的 位 置 。 

由 于 缓冲 器 可 存 n 件 物品 ,因此 ,必须 指出 缓冲 器 中 什么 位 置 已 有 物品 可 供 消费 ,什么 
位 置 沿 无 物品 可 供 生产 者 存放 物品 。 可 以 用 两 个 指针 pp 和 cp 分 别 指示 生产 者 往 缓冲 器 存 
物品 与 消费 者 从 缓冲 器 取 物 品 的 相对 位 置 ,它们 的 初 值 为 ~0”, 生 产 者 和 消费 者 按 顺 序 的 位 
置 去 存 物品 和 取 物 品 。 缓 冲 器 被 循环 使 用 , 即 生产 者 在 缓冲 器 中 顺序 存放 了 件 物品 后 , 则 
以 后 继续 生产 的 物品 仍 从 缓冲 器 的 第 一 个 位 置 开始 存放 。 于 是 ,一 个 生产 者 和 一 个 消费 者 
共享 容量 为 的 缓冲 器 时 ,可 如 下 进行 同步 工作 。 


Buf:array [0..n -1 ] of integer; 
sPdt, sGet : Semaphore; 
pp, cp: integer; 
sPdt := n; sGet :=0;pp:=0; cp:=0; 
cobegin 
procedure producer 
begin 
L1: [produce a product]; 
P (spPdt); 
Buf[ pp ] := product; 
pp:= (pp +1 )modn; 
V(sGet); 
goto Ll1; 
end; 
procedure consumer 
begin 
L2: P (sGet); 
[Take a product from Buf[ cp ] ]; 
cp := (cp + 1)modn; 
V (spdt); 


[consume]; 
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goto L2; 
end; 

coend; 

但 是 ,要 提醒 注意 的 是 ,如 果 PV 操作 使 用 不 当 , 仍 会 出 现 与 时 间 有 关 的 错误 。 例 如 ,有 
个 生产 者 和 g 个 消费 者 ,它们 共享 可 存放 件 物品 的 缓冲 器 ; 为 了 使 它们 能 协调 工作 , 必 
须 使 用 一 个 互 斥 信 号 量 S( 初 值 为 1) ,以 限制 它们 对 缓冲 器 互 斥 地 存 取 。 另 外 ,使 用 两 个 信 
号 量 sPdt( 初 值 为 n) 和 sGet( 初 值 为 0) 来 保证 生产 者 不 往 满 的 缓冲 器 中 存放 物品 ,消费 者 
不 从 空 的 缓冲 器 中 取出 物品 。 同 步 工 作 描 述 如 下 。 


Buf: array [0 . . n —1] of integer; 

sPdt, sGet, S : Semaphore; 

Pp, cp: integer; 

sPdt := n; sGet :=0;pp:=0; cp:=0;S := 1; 


cobegin 
procedure produceri(i = 1, 2,.,p) 
begin 
L1:[produce a product]; 
P (spdt); 
P(S); 


Buf[ pp ] := product; 
PP := (pp + 1)modn; 
V(sGet); 
V(S); 
goto Ll1; 

end; 

procedure consumerj(j = 1,2,.,q) 
begin 
L2:P(sGet); 

P(S); 
[Take a product from Buf[ cp ]]; 
cp := (cp + 1)modn; 
V(sPdt); 
Vv(S); 
[consume]; 
goto L2; 

end 

coend; 


在 这 个 例子 中 ,P 操作 的 顺序 是 很 重要 的 ,如 果 把 生产 者 和 消费 者 进程 中 的 两 个 P 操作 
交换 顺序 , 则 会 导致 错误 。 而 V 操作 的 顺序 却 是 无 关 紧 要 的 。 一 般 来 说 ,用 于 同步 的 信号 
量 上 的 P 操作 先 执行 ,而 用 于 互 斥 的 信号 量 上 的 P 操作 后 执行 。 

生产 者 -消费 者 问题 是 非常 典型 的 问题 ,有 许多 问题 可 归结 为 生产 者 -消费 者 问题 ,但 要 
根据 实际 情况 灵活 运用 。 例 如 , 现 有 4 个 进程 R1、R2、P1、P2, 它 们 共享 可 以 存放 一 个 数 的 
缓冲 器 Buf。 进 程 Rl 每 次 把 来 自 键盘 的 一 个 数 存 入 缓冲 器 Buf 中 , 供 进程 Pl 打印 输出 ; 
进程 R2 每 次 从 磁盘 上 读 一 个 数 存放 到 缓冲 器 Buf 中 , 供 进程 P2 打印 输出 。 为 防止 数据 的 
丢失 和 重复 打印 ,怎样 用 PV 操作 来 协调 这 4 个 进程 的 并 发 执行 ? 


盘 的 
物品 


先 来 分 析 一 下 这 4 个 进程 的 关系 ,进程 R1 和 进程 R2 相当 于 两 个 生产 者 ,接收 来 自 键 


数 或 从 磁盘 上 读 出 的 数 相当 于 这 两 个 进程 各 自生 产 的 物品 。 


两 个 进程 各 自生 产 的 不 同 


要 存 入 共享 的 缓冲 器 Buf 中 ,由 于 Buf 中 每 次 只 能 存 入 一 个 数 ,因此 进程 Rl1 和 进程 R2 


在 存 数 时 必须 互 斥 。 进 程 Pl 和 进程 P2 相当 于 两 个 消费 者 , 它 介 


分 别 消费 进程 RI1 和 进程 


R2 生产 的 物品 。 所 以 进程 R1( 或 进程 R2) 在 把 数 存 入 缓冲 器 Buf 后 应 发 送 消息 通知 进程 
P1( 或 进程 P2)。 进 程 P1( 或 进程 P2) 在 取出 数 之 后 应 发 送 消息 通知 R1( 或 进程 R2) 告 知 组 


冲 器 中 又 允许 放 一 个 新 数 的 消息 。 显 然 , 进 程 R1 与 进程 Pl、 进 


同步 。 
在 分 析 了 进程 之 间 的 关系 后 ,应 考虑 怎样 来 定义 信号 量 。 首 先 ,应 定义 一 个 是 否 允 许 进 
程 Rl 或 进程 R2 把 数 存 入 缓冲 器 的 信号 量 S, 其 初 值 为 "1”。 其 次 ,进程 R1 或 进程 R2 分 别 
要 向 进程 P1 和 进程 P2 发 送 消息 ,应 该 要 有 两 个 信号 量 SI] 和 S2 来 表示 相应 的 消息 , 初 什 
都 应 为 “0”, 表 示 缓 冲 器 中 尚未 有 数 。 至 于 进程 Pl 或 进程 P2 从 缓冲 器 中 取出 数 后 要 发 送 
“缓冲 器 中 允许 放 一 个 新 数 "的 消息 ,这 个 消息 不 应 该 特定 地 发 给 进程 R1 或 进程 R2, 所 以 
只 要 调用 V(S) 就 可 达到 目的 。 到 底 哪个 进程 可 以 把 数 存 入 缓冲 器 中 ,由 进程 Rl 或 进程 
R2 调用 P(S) 来 竞争 。 因 此 ,不 必 再 增加 新 信号 量 了 。 现 定义 三 个 信号 量 ,其 物理 含义 


如 下 


o 


S: 表示 能 否 把 数 存 入 缓冲 器 Buf。 

S1: 表示 缓冲 器 中 是 否 存 有 来 自 键盘 的 数 。 
S2: 表示 缓冲 器 中 是 否 存 有 从 磁盘 上 读 取 的 数 。 
例 3-9 4 个 进程 可 如 下 协调 工作 。 


Buf : integer; 
S, S1, S2 : Semaphore; 
B= 188L IE Oi EE Os 
cobegin 
procedure R1 
x: integer; 
begin 
L1:[ 接 收 来 自 键盘 的 数 ] ; 
x := 接收 的 数 ; 
P(S) ， 
Buf := xi 
V(S1); 
goto Ll1; 
end; 
procedure R2 
Y: integer; 
begin 
I2:[ 从 磁盘 上 读 一 个 数 ]; 
Y := 读 入 的 数 ; 
P(S); 
Buf := y; 
V(S2); 
goto L2; 


程 R2 与 进程 P2 之 间 要 
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end; 
procedure P1 
k: integer; 
begin 
I3: P(S1); 

k:= Buf; 
V(Ss); 
[打印 k 的 值 ]; 
goto L3; 

end; 

procedure P2 
j: integer; 
begin 
L4: P(S2); 

j := Buf 
V(S); 
[打印 j 的 值 ] 
goto L4; 

end; 

coend; 


在 这 里 ,进程 R1 和 进程 R2 在 向 缓冲 器 Buf 中 存 数 之 前 调用 了 P(S) ,其 有 两 个 作用 。 
(1) 由 于 S 的 初 值 为 1”, 所 以 P(S) 限 制 了 每 次 至 多 只 有 一 个 进程 可 以 向 缓冲 器 中 存 


入 一 个 数 ,起 到 了 互 斥 地 向 缓冲 器 中 存 数 的 作用 。 


(2) 当 缓冲 器 中 有 数 且 尚未 被 取 走 时 S 的 值 为 "0”, 当 缓冲 器 中 数 被 取 走 后 S 的 值 又 为 


“1”, 因 此 PCS) 起 到 了 测试 “允许 存 入 一 个 新 数 "的 消息 是 否 到 达 的 同步 作用 


进程 Pl 和 进程 P2 把 需要 的 数 取 走 后 ,都 调用 V(S) 发 出 可 以 存放 一 个 新 数 的 消息 。 


可 见 ,在 这 个 问题 中 信号 量 S 既 被 作为 互 斥 的 信号 量 ,又 被 作为 同步 的 信号 量 。 
在 操作 系统 中 进程 同步 问题 是 非常 重要 的 ,通过 对 一 些 例子 的 分 析 应 该 学 会 怎样 区 别 
进程 的 互 斥 和 进程 的 同步 。PV 操作 是 实现 进程 互 斥 和 进程 同步 的 有 效 工 具 , 但 若 使 用 不 


得 当 则 不 仅 会 降低 系统 效率 而 且 仍 会 产生 错误 ,希望 读者 在 弄 清 PV 操作 作 


用 的 基础 上 , 体 


会 在 各 个 例子 中 调用 不 同 信 号 量 上 的 P 操 作 和 V 操作 的 目的 ,从 而 正确 掌握 对 各 类 问题 的 


解决 方法 。 
3.3.4 读者 - 写 者 问题 
读者 - 写 者 问题 也 是 一 个 经 典 的 并 发 程序 设计 问题 。 有 两 组 并 发 进程 : 


享 一 个 文件 下 ,要 求 : 允许 多 个 读者 同时 对 文件 执行 读 操作 ; 加 只 允许 一 个 


读者 和 写 者 共 
个 写 者 往 文件 中 


写 信息 ; 图 任 一 写 者 在 完成 写 操作 之 前 不 允许 其 他 读者 或 写 者 工作 ; 轩 写 者 执行 写 操作 


前 ,应 让 已 有 的 写 者 和 读者 全 部 退出 。 


于 对 计数 器 rc 操作 的 互 斥 信号 量 ,W 表示 是 否 人 允许 写 的 信号 量 , 于 是 管理 
工作 描述 如 下 。 


单纯 使 用 信号 量 不 能 解决 读者 - 写 者 问题 ,必须 引入 计数 器 rc 记录 读 进程 数 ,rmutex 是 


该 文件 的 同步 


例 3-10 读者 - 写 者 进程 同步 操作 。 


rmutex, W : Semaphore; 
rc: integer; 
rmutex := li;rc := OW := 1; 
cobegin 
procedure readi ( i = 1, 2, …) 
begin 
P (rmutex); 
w= e+ 1 
证 rc == 1 thenP (W); 
V (rmutex); 
[ 读 文件 ]; 
P (rmutex); 
mr = 下 
证 rc ==0 then V (W); 
V (rmutex); 
end; 
procedure writej (j = 1, 2，…) 
begin 
P( W); 
[ 写 文件 ]; 
V (W); 
end; 
coend; 


在 上 面 的 方法 中 ,读者 是 优先 的 。 当 存在 读者 时 , 写 操 作 将 被 延迟 ,并 且 只 要 有 一 个 读 
者 在 访问 文件 ,随后 而 来 的 读者 都 将 被 允许 访问 文件 。 从 而 导致 了 写 进 程 长 时 间 等 待 , 并 有 
可 能 出 现 写 进程 被 * 饿 死 "。 增 加 信号 量 并 修改 上 述 程 序 可 以 得 到 写 进 程 具有 优先 权 的 解决 
方案 能 保证 当 一 个 写 进程 声明 想 写 时 ,不 允许 新 的 读 进程 再 访问 共享 文件 。 

对 于 写 进 程 在 已 有 定义 的 基础 上 还 必须 增加 下 列 信号 量 和 变量 ,引入 计数 器 we 记录 
写 进程 数 ,wmutex 是 用 于 对 计数 器 we 操作 的 互 斥 信号 量 。R 表示 是 否 允许 读 的 信号 , 当 
至 少 有 一 个 写 进程 准备 访问 文件 时 ,用 于 禁止 所 有 的 读 进程 。 

对 于 读 进程 还 需要 一 个 额外 的 信号 量 。 在 R 上 不 允许 建造 长 队列 ,否则 写 进 程 将 不 能 
跳 过 这 个 队列 ,因此 ,只 允许 一 个 读 进程 在 R 上 排队 ,而 所 有 其 他 读 进程 在 等 待 R 之 前 ,在 
信号 量 rlist 上 排队 。 

例 3-11 写 者 优先 ,读者 - 写 者 进程 同步 操作 。 

rmutex, wmutex, rlist, W, R: Semaphore; 


rc, we : integer; 


rmutex := 1; wmutex := 1; rlist := 1;W := 1;R := ljrc := 0; wc := 0; 
cobegin 
procedure readi ( i = 1，2，…) 
begin 
P(rlist); 
P(R); 


P (rmutex); 
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Ee:= ?C4 4 
if rc == 1 thenP (W); 
V (rmutex); 
V(R); 
V(rlist); 
[ 读 文件 ]; 
P (rmutex); 
B= 
if rc == 0 then V (W); 
V (rmutex); 

end; 

procedure writej (j = 1, 2, …) 

begin 
P (wmutex); 
we := wec+ 1; 
if wc == 1 then P(R); 
V(wmutex); 
P( W); 
[ 写 文件 ]; 
V (W); 
P (wmutex); 
Ww := WC— 1 
if wc == 0 then V(R); 
V(wmutex); 

end; 

coend; 


3.3.5 时 间 同 步 问 题 

前 面 讲 到 的 进程 同步 都 属于 空间 上 的 同步 问题 ,其 实 进程 同步 还 有 个 时 间 上 的 同步 问 
题 。 当 一 组 有 关 的 并 发 进程 在 执行 时 间 上 有 严格 的 先后 顺序 时 ,就 会 出 现时 间 上 的 进程 同 
步 问题 。 例 如 ,有 7 个 进程 ,它们 的 执行 顺序 如 图 3-6 所 示 。 


P4 
Pl P3 ps P6 P7 
一 一 一 一 


图 3-6 7 个 进程 的 执行 顺序 


为 了 保证 这 7 个 进程 严格 按照 顺序 执行 ,可 定义 6 个 信号 量 , 其 物理 含义 如 下 。 
S2: 表示 进程 P2 能 否 执行 。 
S3: 表示 进程 P3 能 否 执行 。 
S4: 表示 进程 P4 能 否 执行 。 
S5: 表示 进程 P5 能 否 执行 。 
S6 : 表示 进程 P6 能 否 执行 。 
S7: 表示 进程 P7 能 否 执行 。 


进程 Pl 不 需 定 义 信号 量 , 可 随时 执行 。 这 些 信号 量 的 初 值 为 0, 表 示 不 可 执行 ,而 当 信 
号 量 大 于 等 于 1 时 ,表示 可 执行 。 
例 3-12 进程 执行 顺序 同步 工作 描述 如 下 。 


S2，S3，S4，S5, S5，S7 : Semaphore; procedure P4 


S2:= 0;S3 := 0;S4:= 0; begin 
S5 := 0;S6 := 0;S7 := 0; Pp (S4); 
cobegin 
procedure P1 V (Ss6); 
begin end; 
procedure P5 
V (Ss2); begin 
V (Ss3); P (Ss5); 
V (Ss4); 和 
end V(S6) 
procedure P2 end; 
begin procedure P6 
P (Ss2); begin 
- P (S6); 
V (S57); P (Ss6); 
end; ee 
procedure P3 V(S7); 
begin end; 
P (S93); procedure P7 
可 begin 
V (Ss5); P(S7); 
end; P(S7); 
end; 
coend; 


当 P1 执行 完 后 ,由 于 执行 了 V(S2)、V(S3) 和 V(S4) 三 个 V 操作 ,使 P2、P3 和 P4 在 
P1 后 可 并 发 执行 。P3 执行 完 后 ,由 于 执行 了 V(S5) 操 作 , 则 可 启动 P5 执行 。 而 P6 要 等 
P4 与 P5 两 个 进程 全 部 执行 完 , 执 行 了 两 个 V(S6) 操 作 后 ,才能 启动 执行 。P7 要 等 P2 与 
P6 两 个 进程 全 部 执行 完 , 执 行 了 两 个 V(S7) 操 作 后 才能 启动 执行 。 这 样 , 就 可 以 保证 7 个 
进程 在 时 间 上 的 同步 。 


3.4.1 什么 是 管 程 


信号 量 机 制 为 实现 进程 的 同步 与 互 斥 提供 一 种 原始 、 功 能 强大 且 灵 活 的 工具 ,然而 在 使 
信号 量 和 PV 操作 实现 进程 同步 时 ,对 共享 资源 的 管理 分 散 于 各 个 进程 中 ,进程 能 够 直接 
对 共享 变量 进行 处 理 ,这 样 不 利于 系统 对 临界 资源 的 管理 ,难以 防止 进程 有 意 或 无 意 地 违反 
同步 操作 , 且 容 易 造 成 程序 设计 错误 。 因 此 ,在 进程 共享 主 存 的 前 提 下 ,如 果 能 集中 和 封装 
针对 一 个 共享 资源 的 所 有 访问 并 包括 所 需 的 同步 操作 , 即 把 相关 的 共享 变量 及 其 操作 集中 


进 各 并 发 控制 
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所 作 系 统 原 理 


在 一 起 统一 控制 和 管理 ,就 可 以 方便 地 管理 和 使 用 共享 资源 ,使 并 发 进程 之 间 的 相互 作用 更 
为 清晰 ,也 更 易于 编写 正确 的 并 发 程序 。 

1973 年 ,Hansen 和 Hoare 正式 提出 了 管 程 (Monitor) 的 概念 ,并 对 其 做 了 如 下 的 定义 : 
管 程 是 关于 共享 资源 的 数据 及 在 其 上 的 操作 的 一 组 过 程 或 共享 数据 结构 及 其 规定 的 所 有 操 
作 。 管 程 的 引入 可 以 让 我 们 按 资源 管理 的 观点 ,将 共享 资源 和 一 般 资 源 管理 区 分 开 来 ,使 进 
程 同 步 机 制 的 操作 相对 集中 。 采 用 这 种 方法 ,对 共享 资源 的 管理 可 借助 数据 结构 及 其 上 所 
实施 操作 的 若干 进程 来 进行 ; 对 共享 资源 的 申请 和 释放 可 通过 进程 在 数据 结构 上 的 操作 来 
实现 。 管 程 被 请 求 和 释放 资源 的 进程 所 调用 , 管 程 实质 上 是 把 临界 区 集中 到 抽象 数据 类 型 
模板 中 ,可 作为 程序 设计 语言 的 一 种 结构 成 分 。 对 于 同步 问题 的 解决 , 管 程 和 信号 量具 有 同 
等 的 表达 能 力 。 


3.4.2 使 用 信号 量 的 管 程 


管 程 是 个 或 多 个 过 程 、 一 个 初始 化 序列 和 局 部 数据 组 成 的 软件 模块 ,其 主要 特点 
如 下 。 

(1) 局 部 数据 变量 只 能 被 管 程 的 过 程 访问 ,任何 外 部 过 程 都 不 能 访问 。 

(2) 一 个 进程 通过 调用 管 程 的 一 个 过 程 进入 管 程 。 

(3) 在 任何 时 候 , 只 能 有 一 个 进程 在 管 程 中 执行 ,调用 管 程 的 任何 其 他 进程 都 被 阻塞 ， 
以 等 待 管 程 可 用 。 

前 两 个 特点 让 人 联想 到 面向 对 象 软件 中 对 象 的 特点 。 的 确 ,面向 对 象 操作 系统 或 程序 
设计 语言 可 以 很 容易 地 把 管 程 作为 一 种 具有 特殊 特征 的 对 象 来 实现 。 

通过 给 进程 强加 规定 , 管 程 可 以 提供 一 种 互 斥 机 制 : 管 程 中 的 数据 变量 每 次 只 能 被 一 
个 进程 访问 到 。 因 此 ,可 以 把 一 个 共享 数据 结构 放 在 管 程 中 ,从 而 提供 对 它 的 保护 。 如 果 管 
程 中 的 数据 代表 某 些 资源 ,那么 管 程 为 访问 这 些 资 源 提供 了 互 斥 机 制 。 

为 进行 并 发 处 理 , 管 程 必须 包含 同步 工具 。 例 如 ,假设 一 个 进程 调用 了 管 程 ,并 且 当 它 
在 管 程 中 时 必须 被 阻塞 ,直到 满足 某 些 条 件 。 这 就 需要 一 种 机 制 , 使 得 该 进程 不 仅 被 阻塞 ， 
而 且 能 释放 这 个 管 程 ,以便 某 些 其 他 的 进程 可 以 进入 。 以 后 , 当 条 件 满足 且 管 程 再 次 可 用 时 
需要 恢复 该 进程 并 允许 它 在 阻塞 点 重新 进入 管 程 。 

管 程 通过 使 用 条 件 变量 提供 对 同步 的 支持 ,这 些 条 件 变 量 包含 在 管 程 中 ,并 且 只 有 在 管 
程 中 才能 被 访问 。 有 以 下 两 个 函数 可 以 操作 条 件 变量 。 

(1) cewait(c) : 调用 进程 的 执行 在 条 件 c 上 阻塞 , 管 程 现在 可 被 另 一 个 进程 使 用 。 

(2) csignal(c): 恢复 执行 在 cwait 之 后 因为 某 些 条 件 而 阻塞 的 进程 。 如 果 有 多 个 这 样 
的 进程 ,选择 其 中 一 个 ; 如 果 没 有 这 样 的 进程 ,什么 也 不 做 。 

注意 , 管 程 的 wait 和 signal 操作 与 信号 量 不 同 。 如 果 在 管 程 中 的 一 个 进程 发 信号 ,但 
没有 在 这 个 条 件 变 量 上 等 待 的 任务 , 则 丢弃 这 个 信号 。 
图 3-7 给 出 了 一 个 管 程 的 结构 。 尽 管 一 个 进程 可 以 通过 调用 管 程 的 任何 一 个 过 程 进 入 
管 程 ,但 我 们 仍 可 以 把 管 程 想 象 成 具有 一 个 入 口 点 ,并 保证 一 次 只 有 一 个 进程 可 以 进入 。 其 
他 试图 进入 管 程 的 进程 被 阻塞 并 加 入 等 待 管 程 可 用 的 进程 队列 中 。 当 一 个 进程 在 管 程 中 
时 , 它 可 能 会 通过 发 送 cwait(Cz) 把 自己 暂时 阻塞 在 条 件 x 上 ,随后 它 被 放 入 等 待 条 件 改 变 
以 重新 进入 管 程 的 进程 队列 中 ,在 cwait(z) 调用 的 下 一 条 指令 开始 恢复 执行 。 


待 进 入 管 程 
管 程 等 待 区 域 入 品 的 进程 队列 
条 件 cl 下 本 
cwait(cl) 条 件 变量 
过 程 1 
条 件 cm 下 
cwait(cn) 一 
过 程 人 
紧急 队列 二 
上 sale se le 初始 化 代码 
csignal | 
oan 
出 口 


图 3-7 管 程 的 结构 


如 果 在 管 程 中 执行 的 一 个 进程 发 现 条 件 变 量 x 发 生 了 变化 , 它 将 发 送 csignal(x) ,通知 


相应 的 条 件 队 列 条 件 已 改变 。 


为 给 出 一 个 使 用 管 程 的 例子 ,我 们 再 次 考虑 


生产 者 可 以 通过 管 程 中 的 过 程 put 往 缓 冲 


在 这 个 条 件 上 被 阻塞 。 其 他 某 个 进程 (生产 者 或 消费 者 ) 现 
区 不 再 满 时 ,被 阻塞 进程 可 以 从 队列 中 移出 ,重新 被 激活 ,并 恢复 处 理 。 在 往 缓冲 区 中 放置 


有 界 缓冲 区 的 生产 者 /消费 者 问题 。 例 3-13 
给 出 了 使 用 管 程 的 一 种 解决 方案 , 管 程 模块 PC 控制 着 用 于 保存 和 取 回 物品 的 缓冲 区 , 管 程 
中 有 两 个 条 件 变 量 (使 用 结构 condition 声明 ) : 当 缓冲 区 中 至 少 有 增加 一 个 物品 的 空间 时 ， 
notfull 为 真 ; 当 缓冲 区 中 至 少 有 一 个 物品 时 ,notempty 为 真 。 


区 中 存放 物品 , 它 不 能 直接 访问 buffer。 该 
过 程 首 先 检 查 条 件 notfull, 以 确定 缓冲 区 是 否 还 有 可 用 空间 。 如 


果 没 有 ,执行 管 程 的 进程 


在 可 以 进入 管 程 。 然 后 , 当 缓冲 


一 个 物品 后 ,该 进程 发 送 notempty 条 件 信号 。 对 消费 者 函数 也 可 以 进行 类 似 的 描述 。 
这 个 例子 指出 ,与 信号 量 相 比 较 , 管 程 担负 的 责任 不 同 。 对 于 管 程 , 它 构 造 了 自己 的 互 


斥 机 制 : 生产 者 和 消费 者 不 可 能 同时 访问 缓冲 
csignal 原 语 放 在 管 程 中 ,用 于 防止 进程 往 一 个 满 缓冲 
区 中 取 数 据 项 。 而 在 使 用 信号 量 的 情况 下 ,执行 互 斥 和 同步 都 属于 


例 3-13 PC 管 程 可 描述 如 下 。 


/ * program producer consumer x*/ 
Type PC = monitor 
buffer:array[0..n—-1] of char; 


区 ; 但 是 ,程序 员 必 须 把 适当 的 cwait 和 
区 中 存放 数据 项 ,或 者 从 一 个 空 缓冲 


F 程 序 员 的 责任 。 


/* 分 配 n 个 字符 型 数据 空间 x*/ 


进程 并 发 雁 制 


媳 证 避 


所 作 系 统 原 理 


nextin, nextout : int; /* 缓冲 区 指针 x*/ 
count : int; /* 缓冲 区 中 数据 项 的 个 数 * / 
notfull, notempty : condition; /* 为 同步 设置 的 条 件 变量 * / 
void put (char x) 
{ 

if (count > = n) thencwait(notful1); /* 缓冲 区 满 ,防止 溢出 * / 


buffer[nextin] := x; 
nextin := (nextin + 1) mod n; 
Count++ /* 缓冲 区 中 数据 项 个 数 增 1* / 
csignal (nonempty); /* 释放 任何 一 个 等 待 的 进程 */ 
void take (char x) 
{ 


if (count <= 0) then cwait(not empty); /* 缓冲 区 空 ,防止 下 浇 * / 

x := buffer[nextout]; 

nextout := (nextout + 1) modn; 

count =—; /* 缓冲 区 中 数据 项 个 数 减 1*/ 

csignal (notfull); /* 释放 任何 一 个 等 待 的 进程 * / 
下 
{ 

nextin := 0; nextout := 0; count := 0; /* 缓冲 区 初始 化 为 空 * / 


在 利用 管 程 解决 生产 者 -消费 者 问题 时 ,其 中 的 生产 者 和 消费 者 可 描述 如 下 。 


void producer() 
i 
char x; 
while (true) { 
produce(x); 
PC.put(x); } 
} 


void consumer( ) 
{ 
char x; 
while (true) { 
PC. take(x); 
consume (x); } 


void main() 


parbegin (producer, consumer); 


注意 ,在 上 述 程序 中 ,进程 在 执行 csignal 函数 后 立即 退出 管 程 ,如 果 在 过 程 最 后 没有 发 
生 csignal, Hoare 建议 发 送 该 信号 的 进程 被 阻塞 ,从 而 使 管 程 可 用 ,并 被 放 入 队列 中 直到 管 
程 空闲 。 此 时 一 种 可 能 是 把 阻塞 进程 放置 到 入 口 队 列 中 .这样 它 就 必须 与 其 他 还 没有 进入 
管 程 的 进程 竞争 。 但 是 ,由 于 在 csignal 函数 上 阻塞 的 进程 已 经 在 管 程 中 执行 了 部 分 任务 ， 
因此 使 它们 优先 于 新 进入 的 进程 是 很 有 意义 的 ,这 可 以 通过 建立 一 条 独立 的 紧急 队列 来 实 
现 , 如 图 3-7 所 示 。 


如 果 没 有 进程 在 条 件 z 上 等 待 ,那么 csignal(z) 的 执行 将 不 会 产生 任何 效果 。 而 对 于 
信号 量 , 在 管 程 的 同步 函数 中 可 能 会 产生 错误 。 例如, 如果 省 略 PC 管 程 中 的 任何 一 个 
csignal 函数 ,那么 进入 相应 条 件 队 列 的 进程 将 被 永久 阻塞 。 管 程 优 于 信号 量 之 处 在 于 ,所 
有 的 同步 机 制 都 被 限制 在 管 程 内 部 ,因此 ,不 但 易于 验证 同步 的 正确 性 ,而 且 易 于 检测 出 错 
误 。 此 外 ,如 果 一 个 管 程 被 正确 地 编写 , 则 所 有 进程 对 受 保护 资源 的 访问 都 是 正确 的 ; 而 对 
于 信号 量 ,只 有 当 所 有 访问 资源 的 进程 都 被 正确 地 编写 时 ,资源 访问 才 是 正确 的 。 


3.4.3 使 用 通知 和 广播 的 管 程 


Hoare 关于 管 程 的 定义 要 求 在 条 件 队 列 中 至 少 有 一 个 进程 , 当 另 一 个 进程 为 该 条 件 产 
生 csignal 时 ,该 队列 中 的 一 个 进程 立即 运行 。 因 此 产生 csignal 的 进程 必须 立即 退出 管 程 ， 
或 者 阻塞 在 管 程 上 。 

这 种 方法 有 以 下 两 个 缺陷 。 

(1) 如 果 产 生 csignal 的 进程 在 管 程 内 还 未 结束 , 则 需要 两 个 额外 的 进程 切换 : 阻塞 这 
个 进程 需要 一 次 切换 , 当 管 程 可 用 时 恢复 这 个 进程 又 需要 一 次 切换 。 

(2) 与 信号 相关 的 进程 调度 必须 非常 可 靠 。 产 生 一 个 csignal 时 ,来 自 相应 条 件 队列 中 
的 一 个 进程 必须 立即 被 激活 ,调度 程序 必须 确保 在 激活 前 没有 其 他 进程 进入 管 程 ,否则 , 进 
程 被 激活 的 条 件 又 会 改变 。 例 如 ,在 例 3-13 中 , 当 产 生 一 个 csignal(notempty) 时 ,来 自 
notempty 队列 中 的 一 个 进程 必须 在 一 个 新 消费 者 进入 管 程 之 前 被 激活 。 另 一 个 例子 是 , 生 
产 者 进程 可 能 往 一 个 空 缓冲 区 中 添加 一 个 字符 ,并 在 发 信号 之 前 失败 ,那么 在 notempty 队 
列 中 的 任何 进程 都 将 被 永久 阻塞 。 

Lampson 和 Redell 为 Mesa 语言 开发 了 一 种 不 同 的 管 程 ,他 们 的 方法 克服 了 上 面 列 出 
的 问题 ,并 支持 许多 有 用 的 扩展 , Mesa 管理 结构 还 可 以 用 于 Modula-3 系统 程序 设计 语言 。 
在 Mesa 中 ,csignal 原 语 被 cnotify 取代 ,cnotify 可 解释 如 下 : 当 一 个 正在 管 程 中 的 进程 执 
行 cnotify(Cz) 时 , 它 使 得 之 条 件 队列 得 到 通知 ,但 发 信号 的 进程 继续 执行 。 通 知 的 结果 是 使 
得 位 于 条 件 队列 头 的 进程 在 将 来 合适 的 时 候 且 当 处 理 器 可 用 时 被 恢复 执行 。 但 是 ,由 于 不 
能 保证 在 它 之 前 没有 其 他 进程 进入 管 程 ,因而 这 个 等 待 进程 必须 重新 检查 条 件 。 例 如 ,PC 
管 程 中 的 过 程 现 在 采用 如 例 3-14 所 示 的 代码 。 

让 语句 被 while 循环 取代 ,因此 ,这 个 方案 导致 对 条 件 变量 至 少 多 一 次 额外 的 检测 。 作 
为 回报 , 它 不 再 有 额外 的 进程 切换 ,并 且 对 等 待 进程 在 cnotify 之 后 什么 时 候 运 行 没 有 任何 
限制 。 

与 cnotify 原 语 相 关 的 一 个 很 有 用 的 改进 是 ,给 每 个 条 件 原 语 关联 一 个 监视 计时 器 ,不 
论 条 件 是 否 被 通知 ,一 个 等 待 时 间 超 时 的 进程 将 被 设置 为 就 绪 态 。 当 被 激活 后 ,该 进程 检查 
相关 条 件 ,如 果 条 件 满足 则 继续 执行 。 超 时 可 以 防止 如 下 情况 的 发 生 : 当 某 些 其 他 进程 在 
产生 相关 条 件 的 信号 之 前 失败 时 ,等 待 该 条 件 的 进程 被 无 限制 地 推迟 执行 而 处 于 饥饿 状态 。 

例 3-14 使 用 通知 和 广播 的 PC 管 程 描述 如 下 。 


void put (char x) 

{ 
while (count >= n) cwait (notful1); /x 缓冲 区 满 , 防止 溢出 * / 
buffer[nextin] := x; 


进 穆 并 发 控制 


媳 w 吉 


所 作 系 统 原 理 


nextin := (nextin +1) %$ n; 
Count++; /* 缓冲 区 中 数据 项 个 数 增 1x/ 
cnotify( notempty); /x 通知 正在 等 待 的 进程 * / 


有 
void take (char x) 


while (count <= 0) cwait(notempty); /x* 缓冲 区 空 ,防止 下 溢 * / 

x := buffer[nextout]; 

nextout := (nextout + 1) % ni 

Count —— 3 /* 缓冲 区 中 数据 项 个 数 减 1* / 
cnotify(notfull); /x* 通知 正在 等 待 的 进程 */ 


) 


由 于 进程 是 接 到 通知 而 不 是 强制 激活 的 ,因此 就 可 以 给 指令 表 中 增加 一 条 cbroadcast 
原 语 。 广 播 可 以 使 所 有 的 该 条 件 上 等 待 的 进程 都 被 置 于 就 绪 状 态 , 当 一 个 进程 不 知道 有 多 
少 进程 将 被 激活 时 ,这 种 方式 是 非常 方便 的 。 例 如 ,在 生产 者 、 消 费 者 问题 中 ,假设 put 和 
take 函数 都 适用 于 可 变 长 度 的 字符 块 ,此 时 ,如 果 一 个 生产 者 往 缓冲 区 中 添加 一 批 字 符 , 它 
不 需要 知道 每 个 正在 等 待 的 消费 者 准备 消耗 多 少 字符 ,而 仅 产生 一 个 cbroadcast, 所 有 正在 
等 待 的 进程 都 得 到 通知 并 再 次 尝试 运行 。 

此 外 , 当 一 个 进程 难以 准确 地 判定 将 激活 哪个 进程 时 ,也 可 使 用 广播 。 存 储 管理 程序 就 
是 一 个 很 好 的 例子 。 管 理 程序 有 j 个 空闲 字 节 ,一 个 进程 释放 了 额外 的 & 个 字 节 ,但 它 不 知 
道 哪 个 等 待 进程 一 共 需 要 kk 十 j 个 字 节 ,因此 它 使 用 广播 ,所 有 进程 都 检测 是 否 有 足够 的 存 
储 空间 。 

Lampson/Redell 管 程 优 于 Hoare 管 程 之 处 在 于 ,Lampson/Redell 方法 的 错误 比较 少 。 
在 Lampson/Redell 方法 中 ,由 于 每 个 过 程 在 收 到 信号 后 都 检查 管 程 变 量 , 且 由 于 使 用 了 
while 结构 ,一 个 进程 不 正确 的 广播 会 发 信号 ,不 会 导致 收 到 信号 的 程序 出 错 。 收 到 信号 的 
程序 将 检查 相关 的 变量 ,如 果 期 望 的 条 件 不 满足 , 它 会 继续 等 待 。 

Lampson/Redell 管 程 的 另 一 个 优点 是 , 它 有 助 于 在 程序 结构 中 采用 更 加 模块 化 的 方 
法 。 例 如 ,考虑 一 个 缓冲 区 分 配 程序 的 实现 ,为 了 在 顺序 的 进程 间 合 作 , 必 须 满足 以 下 两 级 
条 件 。 

(1) 保持 一 致 的 数据 结构 。 管 程 强制 实施 互 斥 ,并 允许 对 缓冲 区 的 另 一 个 操作 之 前 完 
成 一 个 输入 或 输出 操作 。 

(2) 在 1 级 条 件 的 基础 上 ,加 上 完成 该 进程 请 求 ,分 配给 该 进程 所 需 的 足够 存储 空间 。 
在 Hoare 管 程 中 ,每 个 信号 传达 1 级 条 件 , 同 时 携带 一 个 隐 含 消息 “我 现在 有 足够 的 
空闲 字 节 ,能 够 满足 特定 的 分 配 请 求 ”", 因 此 该 信号 隐 式 携带 2 级 条 件 。 如 果 后 来 程序 员 改 
变 了 2 级 条 件 的 定义 , 则 需要 重新 编写 所 有 发 信号 的 进程 ; 如 果 程 序 员 改变 了 对 任何 特定 
等 待 进 程 的 假设 (也 就 是 说 ,等 待 一 个 稍微 不 同 的 2 级 不 变量 ), 则 可 能 需要 重新 编写 所 有 发 
信号 的 进程 。 这 样 就 不 是 模块 化 的 结构 ,并 且 当 代码 被 修改 后 可 能 会 引发 同步 错误 (如 被 错 
误 条 件 唤醒 )。 每 当 对 2 级 条 件 做 很 小 的 改动 时 ,程序 员 必 须 记得 去 修改 所 有 的 进程 。 而 对 
于 Lampson/Redell 管 程 ,一 次 广播 可 以 确保 1 级 条 件 并 携带 2 级 条 件 的 线索 ,每 个 进程 将 
自己 检查 2 级 条 件 。 不 论 是 等 待 者 还 是 发 信号 者 对 2 级 条 件 进 行 了 改动 ,由 于 每 个 过 程 都 
会 检查 自己 的 2 级 条 件 , 故 不 会 产生 错误 的 唤醒 。 因 此 ,2 级 条 件 可 以 隐藏 在 每 个 过 程 中 。 


而 对 Hoare 管 程 ,2 级 条 件 必 须 由 等 待 者 带 到 每 个 发 信号 的 进程 的 代码 中 ,这 违反 了 数据 抽 
象 和 进程 间 的 模块 化 原理 。 


3.4.4 用 管 程 解决 哲学 家 进餐 问题 


现在 介绍 如 何 用 管 程 来 解决 哲学 家 进餐 问题 。 在 这 里 ,我 们 认为 哲学 家 可 以 处 在 这 样 
三 种 状态 之 一 : 即 进餐 .饥饿 和 思考 。 相 应 地 ,引入 数据 结构 : 


state: array[0. .4] of (thinking, hungry, eating) 


我 们 为 每 一 位 哲学 家 设置 一 个 条 件 变量 self ( i ) ,每 当 哲学 家 饥饿 但 又 不 能 获得 进餐 
所 需 的 筷子 时 ,他 可 以 执行 cwait(self( i )) 操 作 , 来 推迟 自己 进餐 。 条 件 变 量 可 描述 为 


self : array [0..4] of condition 


在 管 程 中 还 设置 了 以 下 三 个 过 程 。 

(1) pickup( i : 0. .4) 过 程 。 在 哲学 家 进程 中 ,可 利用 该 过 程 去 进餐 。 如 某 哲学 家 处 
于 饥饿 状态 , 且 他 的 左 、 右 两 个 哲学 家 都 未 进餐 时 , 便 允 许 这 位 哲学 家 进餐 ,因为 他 此 时 可 以 
拿 到 左 , 右 两 根 筷子 ; 但 只 要 其 左右 两 位 哲学 家 中 有 一 位 正在 进餐 , 便 不 允许 该 哲学 家 进 
餐 , 此 时 将 执行 cwait(self ( i )) 操 作 来 推迟 进餐 。 

(2) putdown( i: 0. .4) 过 程 。 当 哲学 家 进餐 完毕 ,再 看 他 左右 两 边 的 哲学 家 ,如 果 他 
们 都 处 于 饥饿 状态 且 他 们 左 、 右 两 边 的 哲学 家 都 未 进餐 时 , 便 可 让 他 们 进餐 。 

(3) test( i : 0. .4 ) 过 程 。 该 过 程 为 测试 过 程 ,用 它 去 测试 哲学 家 是 否 已 具备 用 餐 条 
件 , 即 state [kk 十 4mod5] 六 eating and state[ k&] 一 hungry and state[k 十 1mod5] 关 
eating 条 件 为 真 。 若 为 真 . 方 允 许 该 哲学 家 进餐 。 该 过 程 将 被 pickup 和 putdown 两 个 过 程 
调用 。 

例 3-15 于 解决 哲学 家 进餐 问题 的 管 程 描述 如 下 。 


Type dining ~ philosophers = monitor 
state: array[ 0. .4 ] of (thinking, hungry, eating); 
self: array[ 0. . 4] of condition; 
procedure pickup (i : 0 . . 4) 


begin 
state[ i ] := hungry; 
test( i ); 
if state[ i ] != eating then cwait(self[ i ]); 
end; 
procedure putdown (i : 0. . 4) 
begin 


state[ i ] := thinking; 
test(i + 4 mod 5); 
test(i + 1 mod 5); 
end; 
procedure test (k : 0. . 4); 
begin 
if state [k + 4mod5]!= eating 
and state [ k ] = hungry 
and state [k + lmod5 ]!= eating 
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begin 
state[ k ] := eating; 
csignal(self[ k ]); 
end; 
end; 
begin 
fori := 0to4 
do state[ i ] := thinking; 


3.5 进程 间 消 息 传递 


在 计算 机 系统 中 ,并 发 进程 之 间 经 常 要 交换 一 些 信息 。 例 如 ,并 发 进程 间 用 PV 操作 交 
换 信息 实现 进程 的 同步 与 互 斥 , 保 证 安全 地 共享 资源 和 协调 地 完成 任务 。 因 此 ,PV 操作 可 
看 作 是 进程 间 的 一 种 通信 方式 ,但 这 种 通信 只 交换 了 少量 的 信号 ,属于 一 种 低级 通信 方式 。 
有 时 进程 间 要 交换 大 量 的 信息 ,这 种 大 量 信息 的 传递 要 有 专门 的 消息 传递 机 制 来 实现 ,这 是 
一 种 高 级 的 通信 方式 ,也 称 为 进程 通信 ”。 


3.5.1 消息 传递 的 类 型 


随 着 操作 系统 的 发 展 ,进程 通信 机 制 也 在 发 展 ,由 早期 的 低级 进程 通信 机 制 发展 为 能 传 
送 大 量 数据 的 高 级 通信 机 制 。 目 前 ,高 级 通信 机 制 可 归结 为 三 大 类 : 共享 存储 器 系统 、 消 息 
传递 系统 以 及 管道 通信 系统 。 

1. 共享 存储 器 系统 

在 共享 存储 器 系统 中 ,相互 通信 的 进程 共享 某 些 数据 结构 或 共享 存储 区 ,进程 之 间 能 够 
通过 它们 进行 通信 。 由 此 ,又 可 把 它们 进一步 分 成 如 下 两 种 类 型 。 

1) 基于 共享 数据 结构 的 通信 方式 

在 这 种 通信 方式 中 ,要 求 诸 进程 公用 某 些 数据 结构 ,进程 通过 它们 交换 信息 。 如 在 生产 
者 -消费 者 问题 中 ,就 是 把 有 界 缓冲 区 这 种 数据 结构 用 来 实现 通信 。 这 里 ,公用 数据 结构 的 
设置 及 对 进程 间 同 步 的 处 理 都 是 程序 员 的 职责 ,这 无 疑 增加 了 程序 员 的 负担 ,而 操作 系统 却 
只 需 提 供 共享 存储 器 。 因 此 ,这 种 通信 方式 是 低 效 的 ,只 适 于 传递 少量 数据 。 

2) 基于 共享 存储 区 的 通信 方式 

为 了 传输 大 量 数据 ,在 存储 器 中 划 出 了 一 块 共享 存储 区 , 诸 进 程 可 通过 对 共享 存储 区 中 
的 数据 进行 读 或 写 来 实现 通信 。 这 种 通信 方式 属于 高 级 通信 。 进 程 在 通信 前 向 系统 申请 共 
享 存储 区 中 的 一 个 分 区 ,并 指定 该 分 区 的 关键 字 ; 若 系统 已 经 给 其 他 进程 分 配 了 这 样 的 分 
区 , 则 将 该 分 区 的 描述 符 返 回 给 申请 者 。 接 着 ,申请 者 把 获得 的 共享 存储 分 区 连接 到 本 进程 
上 ,此 后 便 可 像 读 / 写 普通 存储 器 一 样 地 读 / 写 公用 存储 分 区 。 

2. 消息 传递 系统 

在 消息 传递 系统 中 ,进程 间 的 数据 交换 以 消息 为 单位 ,程序 员 直 接 利用 系统 提供 的 一 组 
通信 命令 ( 原 语 ) 来 实现 通信 。 操 作 系 统 隐藏 了 通信 的 实现 细节 ,这 大 大 简化 了 通信 程序 编 
制 的 复杂 性 ,因而 获得 广泛 的 应 用 ,并 已 成 为 目前 单机 系统 、 多 机 系统 及 计算 机 网 络 中 的 主 


要 进程 通信 方式 。 消 息 传递 系统 的 通信 方式 属于 高 级 通信 方式 。 根 据 实 现 方式 的 不 同 可 分 
为 直接 传递 方式 和 间接 传递 方式 。 

3. 管道 通信 系统 

所 谓 管道 ,是 指 用 于 连接 一 个 读 进程 和 一 个 写 进 程 ,以 实现 它们 之 间 通 信 的 共享 文件 ， 
又 称 为 pipe 文件 。 向 管道 (共享 文件 ) 提 供 输 入 的 发 送 进程 ( 即 写 进 程 ), 以 字符 流 形式 将 大 
量 的 数据 送 入 管道 ; 而 接收 管道 输出 的 接收 进程 ( 即 读 进程 ), 可 从 管道 中 接收 数据 。 由 于 
发 送 进程 的 接收 进程 是 利用 管道 进行 通信 的 , 故 又 称 为 管道 通信 。 这 种 方式 首创 于 UNIX 
操作 系统 , 因 它 能 传送 大 量 的 数据 且 很 有 效 , 故 又 被 引入 许多 其 他 操作 系统 中 。 

为 了 协调 双方 的 通信 ,管道 通信 机 制 必须 提供 以 下 三 方面 的 协调 能 

(1) 互 斥 。 当 一 个 进程 正 对 pipe 进行 读 / 写 操作 时 , 另 一 个 进程 必须 等 待 。 

(2) 同步 。 当 写 (输入 ) 进 程 把 一 定数 量 数据 写 满 pipe 时 ,应 睡眠 等 待 ,直到 读 ( 输 出 ) 进 
程 取 走 数据 后 ,再 把 它 唤醒 。 当 读 进程 读 一 空 pipe 时 ,也 应 睡眠 等 待 ,直至 写 进程 将 数据 写 
入 管道 后 , 才 将 它 唤醒 。 

(3) 判断 对 方 是 否 存在 。 只 有 确定 对 方 已 存在 时 , 方 能 进行 通信 。 


3.5.2 直接 传递 


直接 传递 是 指 发 送 进程 利用 操作 系统 所 提供 的 发 送 命 令 直 接 把 消息 发 送 给 接收 进程 ， 
而 接收 进程 则 利用 接收 命令 直接 从 发 送 进程 接收 消息 。 在 直接 通信 方式 下 ,企图 发 送 或 接 
收 消息 的 每 个 进程 必须 指出 信件 发 给 谁 或 从 谁 那里 接收 消息 ,可 用 send 原 语 和 receive 原 
语 来 实现 进程 之 间 的 通信 ,这 两 个 原 语 定 义 如 下 。 

(1) send (P, 消 息 ): 把 一 个 消息 发 送 给 进程 P。 

(2) receive (Q ,消息 ) : 从 进程 Q 接收 一 个 消息 。 

这 样 , 进 程 P 和 Q 通过 执行 这 两 个 操作 而 自动 建立 了 一 种 联系 ,并 且 这 种 联系 仅 发 生 
在 这 一 对 进程 之 间 。 消 息 可 以 有 固定 长 度 和 可 变 长 度 两 种 。 固 定 长 度 便于 物理 实现 ,但 使 
程序 设计 增加 困难 ; 而 消息 长 度 可 变 使 程序 设计 变 得 简单 ,但 使 消息 传递 机 制 的 实现 复 
杂 化 。 

我 们 还 可 以 利用 直接 进程 通信 原 语 来 解决 生产 者 -消费 者 问题 。 当 生产 者 生产 出 一 个 
消息 后 ,send 原 语 将 消息 发 送 给 消费 者 进程 ; 而 消费 者 进程 则 利用 receive 原 语 来 得 到 一 个 
消息 。 如 果 消 息 尚 未 生产 出 来 ,消费 者 必须 等 待 ,直到 生产 者 进程 将 消息 发 送 过 来 。 


3.5.3 间接 传递 


采用 间接 传递 方式 时 ,进程 间 发 送 或 接收 消息 通过 一 个 共享 的 数据 结构 一 一 信箱 来 进 
行 ,消息 可 以 被 理解 成 信件 ,每 个 信箱 有 一 个 唯一 的 标识 符 。 当 两 个 以 上 的 进程 有 一 个 共享 
的 信箱 时 ,它们 就 能 进行 通信 。 间 接 通 信 方 式 解除 了 发 送 进程 和 接收 进程 之 间 的 直接 联系 ， 
在 消息 的 使 用 上 灵活 性 较 大 。 间 接 通 信 方 式 中 的 “发 送 ” 和 “接收 ” 原 语 的 形式 如 下 。 

(1) send( A, 消 息 ): 把 一 个 消息 发 送 给 信箱 A。 

(2) receive( A, 消 息 ): 从 信箱 A 接收 一 个 消息 。 

信箱 是 存放 消息 的 存储 区 域 ,每 个 信箱 可 以 分 成 信箱 头 和 信箱 体 两 部 分 。 信 箱 头 指出 
信箱 容量 、 消 息 格式 、 存 放 消息 位 置 的 指针 等 ; 信箱 体 用 来 存放 消息 ,信箱 体 分 成 若干 个 区 ， 
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区 可 容纳 一 个 消息 。 

“发 送 ” 和 “接收 ”两 条 原 语 的 功能 如 下 : 

(1) 发 送 消息 。 如 果 指 定 的 信箱 未 满 , 则 将 消息 送 入 信箱 中 由 指针 所 指示 的 位 置 ,并 释 
放 等 待 该 信箱 中 消息 的 等 待 者 ; 否则 ,发 送 消息 者 被 置 成 等 待 信箱 状态 。 

(2) 接收 消息 。 如 果 指 定 信箱 中 有 消息 , 则 取出 一 个 消息 ,并 释放 等 待 信箱 的 等 待 者 ; 
否则 ,接收 消息 者 被 置 成 等 待 信箱 中 消息 的 状态 。 

两 个 原 语 的 算法 描述 如 下 ,其 中 ,W( ) 和 R( ) 是 让 进程 进入 等 待 队 列 和 出 队列 的 两 个 


Type box = record 


size: integer; {信箱 大 小 } 

count: integer; { 现 有 消息 数 } 

letter: array[ 1 . . n ] of message; {信箱 } 

sl, s2: semaphore; {等 信箱 和 等 消息 信号 量 } 
end; 


procedure send ( B: box, M:message) 
i : integer; 
begin 
if B.count = B. size thenw (B. s1); 
i := B. count + 1; 
B. letter [i] := M; 
B. count := i; 
R(B. s2); 
end; 
procedure receive (B: box, X: message) 
i: integer; 
begin 
if B. count = 0 then W(B. s2); 
B. count := B. count 一 1; 
xX:= B. letter [1 ]; 
if B. countA0 then for i = 1 toB. count do B. letter [i] := B. letter [i+1]; 
R(B. s1); 
end; 


信箱 可 由 操作 系统 创建 ,也 可 由 用 户 进程 创建 ,创建 者 是 信箱 的 拥有 者 。 据 此 ,可 把 信 
箱 分 为 以 下 三 类 。 

(1) 私 用 信箱 。 用 户 进程 可 为 自己 建立 一 个 新 信箱 ,并 作为 该 进程 的 一 部 分 。 信 箱 的 
拥有 者 有 权 从 信箱 中 读 取信 息 ,其 他 用 户 则 只 能 将 自己 构成 的 消息 发 送 到 该 信箱 中 。 这 种 
私 用 信箱 可 采用 单 向 通信 和 链 路 信箱 实现 。 当 拥有 该 信箱 的 进程 结束 时 ,信箱 也 随 之 消失 。 

(2) 公用 信箱 。 它 由 操作 系统 创建 ,并 提供 给 系统 中 的 所 有 核准 进程 使 用 。 核 准 进程 
既 可 把 消息 发 送 到 该 信箱 中 ,也 可 从 信箱 中 取出 发 送 给 自己 的 消息 。 显 然 ,公用 信箱 应 采用 
双向 通信 链 路 的 信箱 来 实现 。 通 常 ,公用 信箱 在 系统 运行 期 间 始终 存在 。 

(3) 共享 信箱 。 它 由 某 进程 创建 ,在 创建 时 或 创建 后 指明 它 是 可 共享 的 ,同时 需 指出 共 
享 进程 (用 户 ) 的 名 字 。 信 箱 的 拥有 者 和 共享 者 都 有 权 从 信箱 中 取 走 发 送 给 自己 的 消息 。 
在 利用 信箱 通信 时 ,在 发 送 进程 和 接收 进程 之 间 存 在 着 下 述 的 4 种 关系 。 
(1) 一 对 一 关系 。 即 可 以 为 发 送 进程 和 接收 进程 建立 一 条 专用 的 通信 链 路 ,使 它们 之 


间 的 交互 不 受 其 他 进程 的 影响 。 
(2) 多 对 一 关系 。 人 允许 提供 服务 的 进程 与 多 个 用 户 进程 之 间 进 行 交互 ,也 称 为 客户 / 服 


务 器 交互 。 
(3) 一 对 多 关系 。 人 允许 一 个 发 送 进程 与 多 个 接收 进程 进行 交互 ,使 发 送 进程 可 用 广播 
形式 向 接收 者 发 送 消息 。 


(4) 多 对 多 关系 。 人 允许 建立 一 个 公用 信箱 ,让 多 个 进程 都 能 向 信箱 中 投递 消息 ,也 可 从 
信箱 中 取 走 属于 自己 的 消息 。 


3.5.4 消息 格式 


消息 的 格式 取决 于 消息 机 制 的 目标 及 该 机 制 是 运行 在 一 台 计 算 机 上 还 是 运行 在 分 布 式 
系统 中 。 对 于 某 些 操作 系统 ,设计 者 优先 选用 短 的 、 固 定 长 度 的 消息 ,以 减 小 处 理 和 存储 的 
开销 。 如 果 需 要 传递 大 量 的 数据 ,数据 可 以 放置 到 一 个 文 辣 


件 中 ,消息 可 以 简单 地 引用 该 文件 。 一 种 更 为 灵活 的 方法 ht 
是 允许 可 变 长 度 的 消息 。 的 
图 3-8 给 出 了 一 种 操作 系统 的 支持 可 变 长 度 消息 的 典 源 进程 标识 符 | 上 消息 类 
型 消息 格式 。 该 消息 被 划分 成 两 部 分 : 包含 相关 信息 的 消 消息 长 度 
息 头 和 包含 实际 内 容 的 消息 体 。 消 息 头 可 以 包含 消息 的 控制 信息 
源 和 目标 的 标识 符 、 长 度 域 .以 及 判定 各 种 消息 类 型 的 类 
型 域 , 还 可 能 含有 一 些 额 外 的 控制 信息 ,例如 ,用 于 创建 消 消息 内 容 三 消息 体 
息 链表 的 指针 域 .记录 源 和 目标 之 间 传 递 的 消息 的 数目 、 
顺序 与 序号 ,以 及 一 个 优先 级 域 。 图 3-8 一 般 消息 格式 


3.5.5 解决 生产 者 -消费 者 问题 


作为 使 用 消息 传递 的 一 个 例子 , 例 3-16 是 解决 有 界 缓冲 区 生产 者 /消费 者 问题 的 一 种 
方法 。 该 例 利 用 了 消息 传递 的 能 力 ,除了 传递 信号 之 外 还 传递 数据 。 它 使 用 了 两 个 信箱 。 
当 生 产 者 产生 了 数据 后 , 它 作为 消息 被 发 送 到 信箱 mayConsume, 只 要 该 信箱 中 有 一 条 消 
息 ,消费 者 就 可 以 开始 消费 。 此 后 mayConsume 用 作 缓 冲 区 ,缓冲 区 中 的 数据 被 组 织 成 消 
息 队列 ,缓冲 区 的 大 小 由 全 变局 变量 capacity 确定 。 信 箱 mayProduce 最 初 填 满 了 空 消 息 ， 
空 消息 的 数量 等 于 信箱 的 容量 ,每 次 生产 使 得 mayProduce 中 的 消息 数 减 少 ,每 次 消费 使 得 
mayProduce 中 的 消息 数 增长 。 

这 种 方法 非常 灵活 ,可 以 有 多 个 生产 者 和 消费 者 ,只 要 它们 都 访问 这 两 个 信箱 即 可 。 系 
统 甚至 可 以 是 分 布 式 系统 ,所 有 生产 者 进程 和 mayProduce 信箱 在 一 个 站 点 上 ,所 有 消费 者 
进程 和 mayConsume 信箱 在 另 一 个 站 点 上 。 

例 3-16 使 用 消息 解决 有 界 缓冲 区 生产 者 /消费 者 问题 。 


const int 
capacity = / * 缓冲 区 容量 */，; 
mall = /x 空 消息 */; 

int i; 

void producer( ) 

{ 


进程 并 发 雁 制 


媳 w 避 


所 作 系 统 原 理 


message pmsg; 
while (true) { 
receive (mayProduce, pmsg) ; 
pmsg = produce ( ); 
send (mayConsume, pmsg) ; 
} 
} 


void consumer ( ) 


小 
message cmsg; 
while (true) { 
receive (mayConsume, cmsg); 
consume (cmsg); 
send (mayProduce, null); 
上 
void main ( ) 
{ 
create mailbox (mayProduce); 
create mailbox (mayConsume); 
for (int i = 1;i<= capacity;i++) send (mayProduce, null); 
parbegin (producer, consumer); 


小 结 


现代 操作 系统 的 核心 是 多 道 程序 设计 、 多 处 理 器 和 分 布 式 处 理 器 ,这 些 方案 的 基础 及 操 
作 系 统 设 计 技 术 的 基础 是 并 发 。 当 多 个 进程 并 发 执行 时 ,无 论 是 在 多 处 理 器 系统 的 情况 下 ， 
还 是 在 单 处 理 器 多 道 程序 系统 中 ,都 会 产生 冲突 和 合作 的 问题 。 

由 于 相关 并 发 进程 在 执行 过 程 中 共享 了 资源 可 能 会 出 现 与 时 间 有 关 的 错误 。 对 涉及 共 
享 资源 的 若干 并 发 进程 的 相关 临界 区 互 斥 执行 ,就 不 会 出 现 与 时 间 有 关 的 错误 。 可 以 采用 
PV 操作 及 管 程 的 方法 来 解决 临界 区 的 互 斥 问题 。 

相互 合作 的 一 组 并 发 进程 ,其 中 每 一 个 进程 都 以 各 自 独 立 的 、 不 可 预知 的 速度 向 前 推 
进 ,但 它们 又 需要 密切 合作 以 实现 一 个 共同 的 任务 .就 需要 解决 进程 同步 问题 。 我 们 仍 可 以 
采用 PV 操作 及 管 程 的 方法 来 解决 进程 同步 问题 。 各 并 发 进程 在 执行 过 程 中 经 常 要 交换 一 
些 信息 ,我 们 把 通过 专门 的 通信 机 制 实现 进程 间 交 换 大 量 信息 的 通信 方式 称 为 “消息 传递 ”， 
也 称 为 “进程 通信 ”。 


第 4 章 内 存 管 理 


内 存 是 计算 机 中 一 种 需要 认真 管理 的 重要 资源 。 就 目前 来 说 ,虽然 一 台 普通 家 用 计算 
机 的 内 存 容量 已 经 是 20 世纪 60 年 代 早 期 全 球 最 大 的 计算 机 IBM 7094 的 内 存 容量 的 一 万 
倍 以 上 ,但 是 程序 大 小 的 增长 速度 比 内 存 容量 的 增长 速度 要 快 得 多 。 正 如 帕 金 森 定律 所 指 
出 的 ;“ 不 管 存储 器 有 多 大 ,程序 都 可 以 把 它 填 满 ”。 

每 个 程序 员 都 梦想 拥有 这 样 的 内 存 : 它 是 私有 的 、 容 量 无 限 大 的 、 速 度 无 限 快 的 ,价格 
低廉 的 ,并 且 是 永久 性 的 存储 器 。 遗 憾 的 是 ,目前 的 技术 还 不 能 为 我 们 提供 这 样 的 内 存 。 

除 此 之 外 的 选择 是 什么 呢 ?” 经 过 多 年 探索 ,人 们 提出 了 “分 层 存储 器 体系 ”的 概念 , 即 在 
这 个 体系 中 ,计算 机 有 若干 兆 字 节 (MB) 快 速 .昂贵 且 易 失 性 的 高 速 缓存 , 数 千 兆 字 节 (GB) 
速度 与 价格 适中 且 同 样 易 失 性 的 内 存 , 以 及 几 兆 兆 字 节 (TB) 低 速 、. 廉 价 、 非 易 失 性 的 磁盘 存 
储 , 另 外 还 有 诸如 DVD 和 USB 等 可 移动 存储 装置 。 操 作 系统 的 工作 是 将 这 个 存储 体系 抽 
象 为 一 个 有 用 的 模型 并 管理 这 个 抽象 模型 。 

操作 系统 中 管理 分 层 存储 体系 的 部 分 称 为 存储 管理 器 。 它 的 任务 是 有 效 地 管理 内 存 ， 
即 记录 哪些 内 存 是 正在 使 用 的 ,哪些 内 存 是 空闲 的 ; 在 进程 需要 时 为 其 分 配 内 存 , 在 进程 使 
用 完 后 释放 内 存 。 


4.0 问题 导入 


在 现代 操作 系统 中 同时 有 多 个 进程 在 运行 ,每 个 进程 的 程序 和 数据 都 需要 放 在 内 存 中 ， 
那么 程序 员 在 编写 程序 时 是 否 需 要 知道 程序 和 数据 的 存放 位 置 呢 ? 如 果 不 知道 ,那么 多 个 
进程 同时 在 内 存 中 运行 ,每 个 进程 应 占用 哪些 空间 呢 ,如何 保证 各 个 进程 占用 的 空间 不 冲突 
呢 ? 内 存 空 间 如何 进 行 分 配 和 管理 呢 ? 


4.1 内 存 管理 概述 


4.1.1 存储 结构 


在 现代 计算 机 系统 中 ,存储 部 件 通常 是 采用 层次 结构 来 组 织 , 以 便 在 成 本 .速度 和 规模 
等 诸 因素 中 获得 较 好 的 性 价 比 。 现 代 通 用 计算 机 的 存储 层次 至 少 应 具有 三 级 : 最 高 层 为 
CPU 寄存 器 ,中 间 层 为 主 存 , 最 底层 为 辅 存 。 在 较 高 档 的 计算 机 中 ,还 可 以 根据 具体 的 功能 
分 工 细 化 为 寄存 器 、 高 速 缓存 . 主 存储 器 、 磁 盘 缓存 磁盘、 可 移动 存储 介质 等 。 如 图 4-1 所 
示 , 在 存储 层次 中 越 往 上 ,存储 介质 的 访问 速度 越 快 ,价格 越 高 ,相对 存储 容量 越 小 。 对 于 不 


握 作 系统 原理 


同 层次 的 存储 介质 ,由 操作 系统 进行 统一 的 管理 。 其 中 ,寄存 器 、 高 速 缓 存 、 主 存储 器 和 磁盘 
缓存 均 属 于 操作 系统 存储 管理 的 管辖 范畴 , 掉 电 后 它们 存储 的 信息 不 再 存在 。 固 定 磁盘 入 
可 移动 存储 介质 属于 设备 管理 的 管辖 范畴 ,它们 存储 的 信息 将 
被 长 期 保存 。 而 磁盘 缓存 本 身 并 不 是 一 种 实际 存在 的 存储 介 

质 , 它 依托 于 固定 磁盘 ,提供 对 主 存储 器 存储 空间 的 扩充 。 


主 存储 迪 主 存储 器 (简称 内 存 或 主 存 ) 是 计算 机 系统 中 的 一 个 主要 部 
件 , 用 于 保存 进程 运行 时 的 程序 和 数据 ,也 称 为 可 执行 存储 器 ， 
目前 其 容量 一 般 为 数 十 MB 到 数 GB, 而 且 容量 还 在 不 断 增加 ， 


而 嵌入 式 计算 机 系统 一 般 仅 有 几 十 KB 到 几 MB。CPU 的 控制 
部 件 只 能 从 主 存 中 取得 指令 和 数据 ,数据 能 够 从 主 存 读 取 并 将 
它们 装 入 到 寄存 器 中 ,或 者 从 寄存 器 存 入 到 主 存 中 。CPU 与 外 
围 设备 之 间 交 换 的 信息 一 般 也 依托 主 存 地 址 空间 。 由 于 主 存 的 访问 速度 远 远 低 于 CPU 的 
执行 速度 ,为 缓和 这 一 矛盾 ,计算 机 系统 引入 了 寄存 器 和 高 速 缓存 。 

寄存 器 访问 速度 最 快 ,完全 能 与 CPU 协调 工作 ,但 价格 昂贵 .容量 小 ,一 般 以 字 (Word) 
为 单位 。 一 个 计算 机 系统 可 能 包括 几 十 个 甚至 上 百 个 寄存 器 ,而 嵌入 式 计 算 机 系统 一 般 仅 
有 几 个 到 几 十 个 寄存 器 ,其 用 于 加 速 存储 访问 速度 ,如 用 寄存 器 存放 操作 数 ,或 用 作 地 址 寄 
存 器 加 快 地 址 转换 速度 。 

高 速 缓存 (Cache) 是 现代 计算 机 结构 中 的 一 个 主要 部 件 , 其 容量 大 于 寄存 器 ,从 几 十 
KB 到 几 MB, 访 问 速度 快 于 主 存 。 


4.1.2 内 存 管理 的 目标 


对 于 管理 来 说 ,我 们 要 问 的 第 一 个 问题 当然 是 内 存 管理 要 达到 的 目标 或 目的 。 

内 存 管 理 就 是 要 提供 一 个 虚幻 的 景象 ,就 像 钱 ,是 虚 的 东西 .实际 上 一 文 不 名 ,但 是 大 家 
认为 它 有 这 个 价值 。 一 个 东西 的 价值 在 于 能 否 满足 我 们 的 渴望 和 需要 。 如 果 能 ,这 个 东西 
就 有 价值 。 内 存 管理 就 是 要 提供 一 个 有 价值 的 虚幻 的 景象 。 用 术语 来 说 就 是 抽象 。 那 么 内 
存 管理 要 提供 哪些 抽象 呢 ? 或 者 说 ,内 存 管理 要 达到 什么 目标 呢 ? 

首先 ,由 于 多 道 程序 同时 存放 在 内 存 中 ,操作 系统 要 保证 它们 之 间 互 不 干扰 。 所 谓 的 互 
不 干扰 就 是 一 个 进程 不 能 随便 访问 另 一 个 进程 的 地 址 空间 。 这 是 内 存 管理 要 达到 的 第 一 个 
目标 。 

那 还 有 没有 别 的 目标 呢 ? 我 们 看 一 下 程序 指令 执行 的 过 程 。 程 序 指令 在 执行 前 加 载 到 
内 存 , 然 后 从 内 存 中 将 一 条 条 指令 读 出 ,执行 相应 的 操作 (从 硬件 层 来 看 ,指令 的 “ 读 取 一 执 
行 循 环 是 计算 机 的 基本 操作 )。 每 条 指令 在 执行 时 需要 读 取 操 作 数 和 写 入 运算 结果 。 要 读 
取 操 作 数 ,就 需要 给 出 操作 数 所 在 的 内 存 地 址 ,这 个 地 址 不 能 是 物理 主 存 地 址 。 这 是 因为 该 
程序 在 何 种 硬件 配置 的 机 器 上 运行 并 不 能 事先 确定 ,操作 系统 自然 不 可 能 对 症 下 药 地 发 出 
对 应 于 某 台 机 器 的 物理 主 存 地 址 。 因 此 ,指令 里 面 的 地 址 是 程序 空间 (虚拟 空间 ) 的 虚拟 地 
址 (程序 地 址 )。 即 程序 发 出 的 地 址 与 具体 机 器 的 物理 主 存 地 址 是 独立 的 。 这 是 内 存 管 理 要 
达到 的 另外 一 个 目标 。 

综 上 所 述 , 内 存 管理 要 达到 如 下 两 个 目标 。 

(1) 地 址 保护 : 一 个 程序 地 址 能 访问 另 一 个 程序 地 址 空间 。 


可 移动 存储 介质 


图 4-1 存储 器 层次 结构 


(2) 地 址 独立 : 程序 发 出 的 地 址 应 与 物理 


E 存 地 址 无 关 。 


这 两 个 目标 就 是 衡量 一 个 内 存 管 理 系统 是 否 完善 的 标准 。 它 是 所 有 内 存 管 理 系统 必须 


提供 的 基本 抽象 。 当 然 ,不同 的 内 存 管 至 


将 在 后 面 论 及 这 些 抽象 时 逐一 说 明 。 


4.1.3 操作 系统 在 内 存 中 的 位 置 


系统 在 此 二 者 之 上 还 提供 了 许多 其 他 抽象 。 本 书 


从 根本 上 来 说 ,计算 机 里 面 运转 的 程序 有 两 种 : 管理 计算 机 的 程序 和 使 用 计算 机 的 程 
序 。 正 如 本 书 前 面 多 次 提 到 ,操作 系统 就 是 管理 计算 机 的 程序 。 而 管理 者 本 身 也 需要 使 用 
资源 。 其 中 的 一 个 资源 就 是 内 存 空间 。 内 存 管 理 的 第 一 个 问题 是 操作 系统 本 身 在 内 存 中 的 
存放 位 置 。 应 该 将 哪 一 部 分 的 内 存 空 间 用 来 存放 操作 系统 呢 ? 或 者 说 ,如 何 将 内 存 空 间 在 
操作 系统 和 用 户 程序 之 间 进 行 分 配 呢 ? 

最 简单 的 方式 就 是 将 内 存 划分 为 上 下 两 个 区 域 ,操作 系统 和 用 户 程序 各 占用 一 个 区 域 ， 


如 图 4-2 所 示 。 
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图 4-2 仅 有 RAM 时 操作 系统 与 用 户 程序 的 内 存 分 配 


比较 起 来 图 4-2(a) 的 构造 最 容易 理解 。 因 为 操作 系统 是 为 用 户 提供 服务 的 ,在 逻辑 上 
处 于 用 户 程序 之 下 。 将 其 置 于 地 址 空间 的 下 面 , 符 合 人 们 的 惯性 思维 。 另 外 ,操作 系统 处 于 
地 址 空间 下 面 还 有 一 个 实际 好 处 : 就 是 在 复位 ,中断 、 陷 入 等 操作 时 ,控制 移交 给 操作 系统 
更 方便 ,因为 操作 系统 的 起 始 地 址 为 0, 无 须 另 行 记 录 操 作 系统 所 处 的 位 置 ,程序 计数 器 清 
零 就 可 以 了 。 清 零 操 作对 于 硬件 来 说 非常 简单 ,无 须 从 总 线 或 寄存 器 读 取 任 何 数据 ; 而 
图 4-2(b) 的 布置 虽然 也 可 以 工作 ,但 显然 与 人 们 习惯 中 操作 系统 在 下 面 的 惯性 思维 不 符 。 


当然 ,除了 上 述 两 种 分 配方 式 外 ,如 果 愿 意 ， 
插 的 分 配方 式 。 只 不 过 这 样 做 没有 半点 儿 好 处 ， 


白 


也 可 以 将 操作 系统 和 用 户 程序 分 拆 , 形 成 穿 
增加 管理 的 复杂 性 。 


由 于 现代 的 计算 机 内 存 除 了 RAM 之 外 ,可 能 还 备 有 ROM。 而 操作 系统 既 可 以 全 部 存 
放 在 ROM 里 ,也 可 以 部 分 存放 在 ROM 里 ,这 样 又 多 出 了 两 种 分 配方 式 ,如 图 4-3 所 示 。 
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图 4-3 备 有 ROM 时 操作 系统 和 用 户 程序 之 内 存 分 配 


内 存 管理 


媳 上 避 


所 作 系 统 原 理 


图 4-3(a) 模 式 下 操作 系统 放 在 ROM 里 面 的 好 处 是 不 容易 被 破坏 ,缺点 是 ROM 要 
做 得 大 ,能 够 容纳 整个 操作 系统 。 由 于 ROM 比较 贵 ,通常 备 有 少量 的 ROM, 只 将 操作 
系统 的 一 部 分 放 在 ROM 里 ,其 余部 分 放 在 RAM 里 ,因此 ,这 两 种 分 配 模式 以 图 4-3(b) 
为 佳 。 
图 4-3(b) 分 配 模式 还 有 另外 一 个 好 处 : 可 以 将 输入 输出 和 内 存 访问 统一 起 来 。 即 将 输 
入 输出 设备 里 面 的 寄存 器 或 其 他 存储 媒介 编 入 内 存 地 址 (在 用 户 程 序 地 址 之 上 ), 使 得 访问 
输入 输出 设备 如 同 访问 内 存 一 样 。 这 种 输入 输出 称 为 内 存 映 射 的 输入 输出 。 如 果 要 访问 的 
地 址 高 于 RAM 的 最 高 地 址 , 则 属于 1/O 操作 ,否则 属于 正常 内 存 操作 。 

这 样 ,根据 操作 系统 是 否 占 用 ROM 或 是 否 采用 内 存 映 射 的 输入 输出 来 分 ,存在 以 下 两 
种 模式 。 

(1) 操作 系统 占用 RAM 的 底层 ,用 户 程序 占用 RAM 的 上 层 。 

(2) 操作 系统 占用 RAM 的 底层 和 位 于 用 户 程 序 地 址 空间 上 面 的 ROM, 用 户 程序 位 于 
中 间 。 

第 二 种 模式 又 分 为 以 下 三 种 情况 。 

(1) 没有 使 用 内 存 映射 的 输入 输出 ,ROM 里 面 全 部 是 操作 系统 。 

(2) 使 用 了 内 存 映 射 的 输入 输出 ,ROM 的 一 部 分 是 操作 系统 , 另 一 部 分 属于 1/O 设备 。 

(3) 使 用 了 内 存 映射 的 输入 输出 ,ROM 全 部 属于 1/O 设备。 

例如 ,CP/M 操作 系统 的 内 存 布局 模式 就 是 上 述 第 一 


种 情况 ,其 BIOS 和 CP/M 内 核 均 处 于 ROM 里 面 ,而 ed 
Shell 和 用 户 程序 处 于 RAM 里 ,如 图 4-4 所 示 。 Ee 
CP/M 是 Control Program for Microcomputers( 微 计 bal 
算 机 控制 程序 ) 的 缩写 , 它 是 一 个 运行 在 Intel 8080 和 a 
Intel 8085 微机 上 的 早期 操作 系统 。 Ox100 
多 数 现代 操作 系统 采用 的 是 第 二 种 模式 : 即 ROM 里 特殊 页面 


包括 操作 系统 的 一 部 分 和 I/O,RAM 里 面 则 包括 操作 
系统 的 其 他 部 分 和 用 户 程序 。Solaris 10 操作 系统 采用 的 ”图 4 CP/M 操作 系统 内 存 布局 
则 是 图 4-3Cb) 模 式 , 即 操作 系统 在 上 面 ,用 户 程序 在 下 面 。 


4.1.4 虚拟 内 存 的 概念 


虚拟 内 存 的 概念 听 上 去 有 点 儿 太 虚拟 ,但 其 实质 则 并 不 难 理解 。 我 们 知道 ,一 个 程序 如 
果 要 运行 ,必须 加 载 到 物理 主 存 中 。 但 是 ,物理 主 存 的 容量 非常 有 限 。 因 此 ,如 果 要 把 一 个 
程序 全 部 加 载 到 物理 主 存 , 则 我 们 所 能 编写 的 程序 将 是 很 小 的 程序 。 它 的 最 大 容量 受制 于 
主 存 容量 (还 要 减 去 操作 系统 所 占 的 空间 和 一 些 临时 缓存 空间 )。 另 外 ,即使 我 们 编写 的 每 
个 程序 的 尺寸 都 小 于 物理 主 存 容 量 , 但 还 是 存在 一 个 问题 : 主 存 能 够 存放 的 程序 数量 将 是 
很 有 限 的 ,而 这 将 极 大 地 限制 多 道 编程 的 发 展 。 

如 何 解决 物理 主 存 容量 偏 小 的 缺陷 呢 ? 最 简单 的 办 法 就 是 购买 更 大 的 物理 主 存 。 而 这 
将 造成 计算 机 成 本 的 大 幅 舰 升 , 可 能 很 多 人 都 会 买 不 起 计算 机 。 

有 没有 办 法 在 不 增加 成 本 的 情况 下 扩大 内 存 容 量 呢 ? 有 ,这 就 是 虚拟 内 存 。 

虚拟 内 存 的 中 心思 想 是 将 物理 主 存 扩大 到 便宜 ,大 容量 的 磁盘 上 ,即将 磁盘 空间 看 作 主 


存 空 间 的 一 部 分 。 用 户 程序 存放 在 磁盘 上 就 相当 于 存放 在 主 存 内 。 用 户 程序 既 可 以 完全 存 
放 在 主 存 , 也 可 以 完全 存放 在 磁盘 上 ,当然 也 可 以 部 分 存放 在 主 存 、 部 分 存放 在 磁盘 。 而 在 
程序 执行 时 ,程序 发 出 的 地 址 到 底 是 在 主 存 还 是 在 磁盘 则 由 操作 系统 的 内 存 管理 模块 负责 
判断 ,并 到 相应 的 地 方 进行 读 写 操作 。 事 实 上 ,可 以 更 进一步 ,将 缓存 和 磁带 也 包括 进来 , 构 
成 一 个 效率 、 价 格 、 容 量 错落 有 致 的 存储 架构 。 即 虚拟 内 存 要 提供 的 就 是 一 个 空间 像 磁盘 那 
样 大 、 速 度 像 缓 存 那 样 高 的 主 存储 系统 ,如 图 4-5 所 示 。 而 对 程序 地 址 所 在 位 置 (缓存 、 主 存 
和 磁盘 ) 的 判断 则 是 内 存 管理 系统 的 一 个 中 心 功能 。 


所 香 应 用 软件 全 绩 虚拟 机 界面 


| ] | 虚拟 内 存 
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图 4-5 虚拟 内 存 所 提供 的 抽象 


值得 特别 指出 的 是 ,虚拟 内 存 是 操作 系统 发 展 史 上 的 一 个 革命 性 突破 (当然 , 它 也 是 使 
操作 系统 变 得 更 加 复杂 的 一 个 主要 因素 )。 因 为 有 了 虚拟 内 存 , 我 们 编写 的 程序 从 此 不 再 受 
尺寸 的 限制 (当然 还 受制 于 虚 地 址 空间 大 小 的 限制 )。 

虚拟 内 存 除了 让 程序 员 感 觉 到 内 存 容 量 大 大 增加 之 外 ,还 让 程序 员 感 觉 到 内 存 速 度 也 
加 快 了 。 这 是 因为 虚拟 内 存 将 尽 可 能 从 缓存 满足 用 户 访问 请 求 , 从 而 给 人 以 速度 提升 了 的 
感觉 。 从 这 个 角度 来 看 ,虚拟 内 存 就 是 实际 存储 架构 与 程序 员 对 内 存 的 要 求 之 间 的 一 座 

当然 ,容量 增 大 也 好 ,速度 提升 也 好 ,都 是 虚拟 内 存 提供 的 一 个 幻象 ,实际 上 并 不 是 这 么 
可 事 儿 ,但 用 户 感觉 到 是 真 的 ,这 就 是 魔术 。 操 作 系统 的 一 个 角色 就 是 魔术 师 。 


4.2 内 存 管理 的 基础 


当 研究 与 内 存 管 理 相关 的 各 种 机 制 和 策略 时 ,清楚 内 存 管理 的 基础 内 容 是 非常 有 用 的 。 
内 存 管理 的 基础 需求 如 下 。 

(1) 重 定位 。 

(2) 保护 。 

(3) 共享 。 

(4) 逻辑 组 织 。 

(5) 物理 组 织 。 
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4.2.1 重 定位 


在 多 道 程序 设计 系统 中 ,可 用 的 内 存 空间 通常 被 多 个 进程 共享 。 通 常情 况 下 ,程序 员 并 
不 能 事先 知道 在 某 个 程序 执行 期 间 会 有 其 他 哪些 程序 驻 留 在 内 存 中 。 此 外 还 希望 通过 提 
供 一 个 巨大 的 就 绪 进 程 池 ,能 够 把 活动 进程 换 入 或 换 出 内 存 , 以 便 使 处 理 器 的 利用 率 最 
大 化 。 一 旦 程序 被 换 出 到 磁盘 ,当下 一 次 被 换 入 时 ,如 果 必 须 放 在 和 被 换 入 前 相同 的 内 
存 区 域 ,那么 这 将 会 是 一 个 很 大 的 限制 。 为 了 避免 这 种 限制 ,需要 把 进程 重 定位 到 内 存 
的 不 同 区 域 。 
因此 ,我 们 事先 不 知道 程序 将 会 被 放置 到 哪个 区 域 , 并 且 我 们 必须 允许 程序 通过 交换 技 
术 (Swapping) 在 内 存 中 移动 。 这 关系 到 一 些 与 寻 址 相关 的 技术 问题 ,如 图 4-6 所 示 。 该 图 
描述 了 一 个 进程 映像 。 为 简单 起 见 , 假 设 该 进程 映像 占据 了 内 存 中 的 一 段 相 邻 的 区 域 。 显 
然 ,操作 系统 需要 知道 进程 控制 信息 和 执行 栈 的 位 置 ,以 及 该 进程 开始 执行 程序 的 入 口 点 。 
于 操作 系统 管理 内 存 并 负责 把 进程 放 入 内 存 , 因 此 可 以 很 容易 地 访问 到 这 些 地 址 。 此 外 ， 
处 理 器 必须 处 理 程序 内 部 的 内 存 访问 。 跳 转 指令 包含 下 一 步 将 要 执行 的 指令 的 地 址 ,数据 
访问 指令 包含 被 访问 数据 的 字 节 或 字 的 地 址 。 处 理 器 硬件 和 操作 系统 软件 必须 能 够 通过 某 
种 方式 把 程序 代码 中 的 内 存 访问 转换 成 实际 的 物理 内 存 地 址 ,并 反映 程序 在 内 存 中 的 当前 
位 置 。 


进程 控制 信息 
进程 控制 志 
跳 转 指令 超 训 
[人 地 址 值 增加 
数据 访问 
一 一 当前 栈 顶 
本 


图 4-6 进程 在 寻 址 方面 的 需求 


4.2.2 保护 与 共享 


每 个 进程 都 应 该 受到 保护 ,以 免 被 其 他 进程 有 意 或 无 意 地 干涉 。 因 此 ,该 进程 以 外 的 其 
他 进程 中 的 程序 不 能 未 经 授权 地 访问 (进行 读 操作 或 写 操作 ) 该 进程 的 内 存单 元 。 在 某 种 意 
义 上 ,要 满足 重 定位 的 需求 增加 了 满足 保护 需求 的 难度 。 由 于 程序 在 内 在 中 的 位 置 是 不 可 
预测 的 ,因而 在 编译 时 不 可 能 检查 绝对 地 址 来 确保 保护 。 并 且 , 大 多 数 程序 设计 语言 允许 在 
运行 时 进行 地 址 的 动态 计算 (例如 ,通过 计算 数组 下 标 或 数据 结构 中 的 指针 )。 因 此 ,必须 在 
运行 时 检查 进程 产生 的 所 有 内 存 访问 ,以 确保 它们 只 访问 了 分 配给 该 进程 的 内 存 空间 。 幸 
运 的 是 , 既 支持 重 定位 也 支持 保护 需求 的 机 制 已 经 存在 。 

通常 ,用 户 进程 不 能 访问 操作 系统 的 任何 部 分 ,不 论 是 程序 还 是 数据 。 并 且 , 一 个 进程 
中 的 程序 通常 不 能 跳 转 到 另 一 个 进程 中 的 指令 。 如 果 没 有 特别 的 许可 ,一 个 进程 中 的 程序 
不 能 访问 其 他 进程 的 数据 区 。 处 理 器 必须 能 够 在 执行 时 终止 这 样 的 指令 。 


注意 ,内 存 保护 的 需求 必须 由 处 理 器 (硬件 ) 来 满足 ,而 不 是 由 操作 系统 (软件 ) 来 满足 。 
这 是 因为 操作 系统 不 能 预测 程序 可 能 产生 的 所 有 内 存 访问 ; 即使 可 以 预测 ,提前 审查 每 
个 进程 中 可 能 存在 的 内 存 违法 访问 也 是 非常 费时 的 。 因 此 ,只 能 在 指令 访问 内 存 时 来 判 
断 这 个 内 存 访问 是 否 违法 ( 存 取 数据 或 跳 转 )。 为 实现 这 一 点 ,处 理 器 硬件 必须 具有 这 个 
能 力 。 

任何 保护 机 制 都 必须 具有 一 定 的 灵活 性 ,以 允许 多 个 进程 访问 内 存 的 同一 部 分 。 例 如 ， 
如 果 多 个 进程 正在 执行 同一 个 程序 , 则 允许 每 个 进程 访问 该 程序 的 同一 个 副本 要 比 让 每 个 
进程 有 自己 单独 的 副本 更 有 优势 。 合 作 完 成 同一 个 任务 的 进程 可 能 需要 共享 访问 相同 的 数 
据 结构 。 因 此 内 存 管理 系统 必须 允许 对 内 存 共享 区 域 进行 受 控 访 问 ,而 不 会 损害 基本 的 保 
护 。 我 们 将 会 看 到 用 于 支持 重 定位 的 机 制 也 支持 共享 。 


4.2.3 ”逻辑 组 织 


计算 机 系统 中 的 内 存 总 是 被 组 织 成 线性 的 (或 一 维 的 ) 地 址 空间 ,并 且 地 址 空间 是 
系列 字 节 或 字 组 成 的 。 外 部 存储 器 (简称 外 存 ) 在 物理 层 上 也 是 按 类 似 方式 组 织 的 ,尽管 这 
种 组 织 方式 类 似 于 实际 的 机 器 硬件 ,但 它 并 不 符合 程序 构造 的 典型 方法 。 大 多 数 程 序 被 组 
织 成 模块 , 某 些 模块 是 不 可 修改 的 (只 读 、 只 执行 ), 某 些 模块 包含 可 以 修改 的 数据 。 如 果 操 
作 系 统 和 计算 机 硬件 能 够 有 效 地 处 理 以 某 种 模块 形式 组 织 的 用 户 程序 和 数据 , 则 会 带 来 很 
多 好 处 。 

(1) 可 以 独立 地 编写 和 编译 模块 ,系统 在 运行 时 解析 从 一 个 模块 到 其 他 模块 的 所 有 
引用 。 

(2) 通过 适度 的 额外 开销 ,可 以 给 不 同 的 模块 以 不 同 的 保护 级 别 (只 读 、 只 执行 )。 

(3) 可 以 引入 某 种 机 制 , 使 得 模块 可 以 被 多 个 进程 共享 。 在 模块 级 提供 共享 的 优点 在 
于 , 它 符合 用 户 看 待 问 题 的 方式 ,因此 用 户 也 可 以 很 容易 地 指定 需要 的 共享 。 

最 易于 满足 这 些 需 求 的 工具 是 分 段 , 这 也 是 第 5 章 要 探讨 的 一 种 内 存 管理 技术 。 


4.2.4 物理 组 织 


正如 4.1.1 节 所 论述 的 ,计算 机 存储 器 至 少 要 被 组 织 成 两 级 , 称 为 内 存 和 外 存 。 内 存 提 
供 快速 的 访问 ,成 本 也 相对 比较 高 .并且 内 存 是 易 失 性 的 , 即 它 不 能 提供 永久 存储 。 外 存 比 
内 存 慢 而 且 便宜 , 它 通常 是 非 易 失 性 的 。 因 此 ,大 容量 的 外 存 可 以 用 于 长 期 存储 程序 和 数 
据 ,而 较 小 的 内 存 则 用 于 保存 当前 使 用 的 程序 和 数据 。 

在 这 种 两 级 方案 中 ,系统 主要 关注 的 是 内 存 和 外 存 之 间 信 息 流 的 组 织 。 可 以 让 程序 员 
负责 组 织 这 个 信息 流 , 但 由 于 以 下 两 方面 的 原因 ,这 种 方式 是 不 切实 际 的 ,也 是 不 合乎 要 
求 的 。 

(1) 可 供 程序 和 数据 使 用 的 内 存 可 能 不 足 。 在 这 种 情况 下 ,程序 员 必 须 采 用 覆盖 
(Overlaying) 技 术 来 组 织 程序 和 数据 。 不 同 的 模块 被 分 配 到 内 存 中 的 同一 块 区 域 , 主 程序 
负责 在 需要 时 换 入 或 换 出 模块 。 即 使 有 编译 工具 的 帮助 ,覆盖 技术 的 实现 仍然 非常 浪费 程 
序 员 的 时 间 。 

(2) 在 多 道 程 序 设计 环境 中 ,程序 员 在 编写 代码 时 并 不 能 知道 可 用 空间 的 大 小 及 位 置 。 


媳 上 避 
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显然 ,在 两 级 存储 器 间 移 动 信息 的 任务 应 该 是 一 种 系统 责任 ,而 该 任务 恰恰 就 是 存储 管 
理 的 本 质 所 在 。 


4.3 单 道 编程 中 的 内 存 管 理 


最 简单 的 内 存 管理 是 单 道 程序 下 的 内 存 管 理 。 在 单 道 编程 环境 下 ,整个 内 存 里 面 只 有 
两 个 程序 : 一 个 是 用 户 程序 , 另 一 个 是 操作 系统 。 由 于 只 有 一 个 用 户 程序 ,而 操作 系统 所 占 
里 的 内 存 空间 是 恒定 的 ,我 们 可 以 将 用 户 程序 总 是 加 载 到 同一 个 内 存 地 址 上 。 即 用 户 程序 
永远 从 同一 个 地 方 开始 执行 。 在 这 种 管理 方式 下 ,操作 系统 永远 跳 转 到 同一 个 地 方 来 启动 
用 户 程序 。 这 样 , 用 户 程序 里 面 的 地 址 都 可 以 事先 计算 出 来 , 即 在 程序 运行 前 就 计算 出 所 有 
的 物理 地 址 。 这 种 在 运行 前 即将 物理 地 址 计算 好 的 方式 叫 作 静态 地 址 翻译 。 
这 种 内 存 管 理 方式 是 如 何 达到 内 存 管 理 的 两 个 目的 的 呢 ? 首先 看 地 址 独立 。 固 定 地 址 
的 内 存 管理 达到 地 址 独立 了 吗 ? 那 就 看 看 用 户 在 编写 程序 时 是 否 需要 知道 该 程序 将 要 运行 
的 物理 内 存 知识 。 显 然 不 需要 。 因 而 用 户 在 编程 时 用 的 虚 地 址 无 须 考虑 具体 的 物理 内 存 ， 
即 该 管理 模式 达到 了 地 址 独立 。 那 么 它 是 如 何 达到 目的 的 呢 ? 办 法 就 是 将 用 户 程序 加 载 到 
同一 个 物理 地 址 上 。 通 过 静态 编译 即 可 完成 虚 地 址 到 物理 地 址 的 映射 ,而 这 个 静态 翻译 工 
作 可 以 由 编译 器 或 者 加 载 器 来 实现 。 

那么 内 存 管理 的 另 一 抽象 , 即 地 址 保护 ,达到 了 吗 ? 那 要 看 该 进程 是 否 会 访问 到 别 的 用 
户 进程 空间 ,或 者 别 的 用 户 进程 是 否 会 访问 该 进程 地 址 。 答 案 是 不 会 。 因 为 整个 系统 里 面 
只 有 一 个 用 户 程 序 。 因 此 ,固定 地 址 的 内 存 管理 因为 只 运行 一 个 用 户 程 序 而 达到 地 址 保护 。 

固定 地 址 的 内 存 管 理 单元 非常 简单 ,实际 上 并 不 需要 任何 内 存 管理 单元 。 因 为 程序 发 
出 的 地 址 已 经 是 物理 地 址 ,在 执行 过 程 中 无 须 进 行 任 何 地 址 翻译 。 而 这 种 情况 的 直接 结果 
就 是 程序 运行 速度 快 ,因为 越过 了 地 址 翻译 这 个 步骤 。 

当然 ,固定 地 址 的 内 存 管理 其 缺点 也 很 明显 : @ 整 个 程序 要 加 载 到 内 存 空 间 中 去 。 这 
样 将 导致 比 物理 内 存 大 的 程序 无 法 运行 。@ 只 运行 一 个 程序 造成 资源 浪费 。 如 果 一 个 程序 
很 小 ,虽然 所 用 内 存 空间 小 ,但 剩 下 的 内 存 空 间 也 无 法 使 用 。@@ 可 能 无 法 在 不 同 的 操作 系 
统 下 运行 ,因为 不 同 操作 系统 占用 的 内 存 空间 大 小 可 能 不 一 样 ,使 得 用 户 程序 的 起 始 地 
址 可 能 不 一 样 。 这 样 在 一 个 系统 环境 下 编译 出 来 的 程序 很 可 能 无 法 在 另 一 个 系统 环境 
下 执行 。 


4.4 多 道 编 程 中 的 内 存 管理 


单 道 编程 的 缺点 前 面 已 阐述 过 ,为 了 克服 单 道 编程 的 缺点 ,我 们 发 明了 多 道 编程 。 随 着 
多 道 编程 度数 的 增加 ,CPU 和 内 存 的 利用 效率 也 随 着 增加 。 当 然 , 这 种 增加 有 个 限度 ,超过 
这 个 限度 , 则 因为 多 道 程序 之 间 的 资源 竞争 反而 造成 系统 效率 下 降 。 

虽然 多 道 编 程 可 以 极 大 地 改善 CPU 和 内 存 的 效率 ,改善 用 户 响应 时 间 , 但 是 天 下 没有 
免费 的 午餐 ,这 种 效率 和 响应 时 间 的 改善 是 需要 付出 代价 的 。 

这 个 代价 是 什么 呢 ? 当然 是 操作 系统 的 复杂 性 。 因 为 多 道 编程 的 情况 下 ,无 法 将 程序 
总 是 加 到 固定 的 内 存 地 址 上 ,也 就 是 无 法 使 用 静态 地 址 翻译 。 这 样 我 们 就 必须 在 程序 加 载 


完毕 后 才能 计算 物理 地 址 ,也 就 是 在 程序 运行 时 进行 地 址 翻译 ,这 种 翻译 称 为 动态 地 址 翻 


译 。 图 4-7 描述 的 就 是 动态 地 址 翻译 的 示意 图 。 


虚拟 地 址 .入 物理 地 址 


也 许 


了 吗 ? 但 
做 还 将 带 
可 以 访问 


图 4-7 动态 地 址 翻译 


有 读者 会 想 ,我 们 可 以 在 内 存 固 定 几 个 地 址 出 来 ,比如 说 4 个 。 这 样 就 可 以 同时 加 
载 4 个 程序 到 内 存 , 而 这 4 个 程序 分 别 加 到 这 4 个 固定 的 地 址 ,不 就 可 以 进行 静态 地 址 翻译 


是 谁 能 提前 知道 某 个 特定 的 程序 将 加 载 到 4 个 固定 地 址 的 哪 


一 个 呢 ? 而 且 , 这 样 


来 地 址 保护 上 的 困难 。 既 然 所 有 的 程序 皆 发 出 物理 地 址 ,该 地 址 是 否 属于 该 程序 


的 空间 将 无 法 确认 。 这 样 ,程序 之 间 的 互相 保护 就 成 了 问题 。 


多 道 编程 的 内 存 管理 是 如 何 进 行动 态 地 址 翻译 的 呢 ? 那 得 看 内 存 管理 的 策略 。 多 道 编 


程 下 的 内 
4.4.1 


顾 名 
下 面 的 分 


存 管理 策略 有 两 种 : 固定 分 区 和 非 固定 分 区 。 
固定 分 区 的 多 道 编程 内 存 管 理 


思 义 ,固定 分 区 的 管理 就 是 将 内 存 分 为 固定 的 几 个 区 域 .每 个 区 域 的 大 小 固定 。 最 


一 样 。 考 
选择 一 个 


虑 到 程序 大 小 不 一 的 实际 情况 ,分 区 的 大 小 通常 也 各 不 相同 。 
当前 闲置 且 容 量 够 大 的 分 区 进行 加 载 ,如 图 4-8 所 示 。 


OxFFFF... 


分 区 n 


a 用 户 程序 


操作 系统 


图 4-8 共享 队列 的 固定 分 区 


分 区 2 | 国 


区 为 操作 系统 占用 ,其 他 分 区 由 用 户 程序 使 用 。 这 些 分 区 大 小 可 以 一 样 , 也 可 以 不 


当 需 要 加 载 程序 时 ， 


在 这 种 模式 下 , 当 一 个 新 的 程序 想 要 运行 ,必须 排 在 一 个 共同 的 队列 里 等 待 。 当 有 空闲 
分 区 时 ,才能 进行 加 载 。 由 于 程序 大 小 和 分 区 大 小 不 一 定 匹配 ,有 可 能 天 


一 站 大 办 


EB 成 一 个 小 程序 占用 


区 的 情况 ,从 而 造成 内 存 里 虽然 有 小 分 区 闲置 ,但 无 法 加 载 大 程序 的 情况 。 如 果 在 


前 面 加 载 小 程序 时 考虑 到 这 一 点 ,可 以 将 小 程序 加 载 到 小 分 区 里 ,就 不 会 出 现 这 种 情况 (或 


者 说 至 少 
从 区 一 
这 样 


降低 这 种 情况 发 生 的 概率 )。 这 样 ,我 们 就 会 想到 也 许可 以 采 
个 队列 。 程 序 按 大 小 排 在 相应 的 队列 里 ,如 图 4-9 所 示 。 


多 个 队列 , 即 给 每 


不 同 的 程序 有 不 同 的 队列 ,就 像 在 社会 中 不 同 的 社会 阶层 有 不 同 的 待遇 一 样 。 当 
然 , 这 种 方式 也 有 缺点 ,就 是 如 果 还 有 空闲 分 区 ,但 等 待 的 程序 不 在 该 分 
将 造成 有 空间 而 不 能 运行 程序 的 尴 估 处境 。 


区 的 等 待 队列 上 ,就 


内 存 管理 


媳 上 站 
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分 区 2 
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图 4-9 分 开 队列 的 固定 分 区 


4.4.2 地址 翻译 的 方法 


在 多 道 编程 环境 下 ,由 于 程序 加 载 到 内 存 的 地 址 不 是 固定 的 (有 多 个 地 方 可 加 载 ) ,必须 
对 地 址 进行 翻译 。 那 么 如 何 来 翻译 呢 ? 

我 们 看 到 一 个 程序 是 加 载 到 内 存 里 事先 划分 好 的 某 片区 域 ,而 且 该 程序 是 整个 加 载 进 
去 。 该 程序 里 面 的 虚 地 址 只 要 加 上 其 所 占 区 域 的 起 始 地 址 即 可 获得 物理 地 址 。 因 此 ,翻译 
过 程 非常 简单 : 


物理 地 址 一 虚拟 地 址 十 程序 所 在 区 域 的 起 始 地 址 

程序 所 在 区 域 的 起 始 地 址 称 为 (程序 ) 基 址 。 

另外 ,由 于 有 多 个 程序 在 内 存 空 间 中 ,需要 进行 地 址 保护 。 由 于 每 个 程序 占用 连续 的 一 
片 内 存 空间 ,因此 只 要 其 访问 的 地 址 不 超出 该 片 连续 空间 , 则 为 合法 访问 。 因 此 ,地 址 保护 
也 变 得 非常 简单 ,只 要 访问 的 地 址 满足 下 面 的 条 件 即 为 合法 访问 。 

程序 所 在 区 域 的 起 始 地 址 三 有 效 地 址 夸 程 序 所 在 区 域 的 起 始 地 址 十 程序 长 度 

由 此 可 见 , 只 需要 设置 两 个 端 值 : 基 址 和 极限 , 即 可 达到 地 址 翻译 和 地 址 保护 的 目的 。 
这 两 个 端 值 可 以 由 两 个 寄存 器 来 存放 ,分 别称 为 基 址 寄存 器 和 极限 寄存 器 。 在 固定 分 区 下 ， 
基 址 就 是 固定 内 存 分 区 中 各 个 区 域 的 起 始 内 存 地 址 ,而 极限 则 是 所 加 载 程序 的 长 度 ( 记 住 ， 
不 是 内 存 各 个 分 区 的 上 限 )。 

这 样 ,每 次 程序 发 出 的 地 址 需要 跟 极 限 比 较 大 小 ; 如 果 大 于 极限 , 则 属 非 法 访问 。 在 这 
个 时 候 将 陷入 内 核 , 终 止 进程 (在 个 别 操作 系统 上 ,也 可 能 进入 一 个 异常 处 理 的 过 程 ); 否则 
将 基 址 加 上 偏 移 获 得 物理 地 址 ,就 可 以 合法 地 访问 这 个 物理 地 址 。 


Xx| 


证 (虚拟 地 址 > 极限 ) { 

陷入 内 核 

终止 进程 ( core dump ) 
} else{ 

物理 地 址 = 虚拟 地 址 + 基 址 
} 


由 此 可 见 ,实现 基 址 极限 管理 的 硬件 也 很 简单 : 一 个 加 法 器 和 一 个 比较 器 即 可 。 这 样 ， 
对 每 个 程序 来 说 , 它 仿佛 独占 了 一 个 内 存 空 间 从 0 到 极限 的 计算 机 ,如 图 4-10 所 示 。 


OxFFFF... 
用 户 程序 
粮 限 [一 一 一 一 一 一 ~ 基 址 + 极限 
i 基 址 
虚拟 地 址 
0 
物理 内 存 


图 4-10 基 址 极限 的 概念 


那么 怎么 知道 一 个 程序 有 多 大 了 呢 ? 编译 过 后 ,我 们 就 可 以 得 到 这 个 程序 的 大 小 。 基 址 
和 极限 是 很 重要 的 两 个 参数 ,只 有 内 核能 够 改变 它们 。 如 果 要 切换 程序 ,只 需 将 保存 基 址 和 
极限 寄存 器 的 值 按照 新 程序 的 情况 重新 设置 即 可 。 


4.4.3 动态 地 址 翻译 的 优点 


动态 地 址 翻译 虽然 增加 了 系统 消耗 ,不 如 静态 地 址 翻译 效率 高 ,但 其 带 来 的 优点 远 远 超 
过 静态 地 址 翻译 。 第 一 个 优点 是 灵活 。 因 为 实施 了 动态 地 址 翻译 ,就 无 须 依赖 编译 器 或 加 
载 器 来 进行 静态 地 址 翻译 ,可 以 将 程序 随便 加 载 到 任何 地 方 , 极 大 地 提高 操作 系统 操作 的 灵 
活性 。 第 二 个 优点 是 , 它 是 实施 地 址 保护 的 “不 二 法 门 ”。 要 想 进 行 地址 保护 ,就 必须 对 每 个 
访问 地 址 进行 检查 ,而 动态 地 址 翻译 恰恰 就 能 做 到 这 一 点 。 第 三 个 优点 则 更 为 重要 , 它 使 虚 
拟 内 存 的 概念 得 以 实现 。 虚 拟 内 存 就 是 子虚乌有 的 内 存 , 这 个 内 存 的 空间 可 以 非常 大 , 比 物 
理 空间 大 很 多 。 那 么 虚拟 内 存 的 根本 是 将 内 存 扩展 到 磁盘 上 ,就 是 将 磁盘 也 当成 内 存 的 一 
部 分 。 从 这 里 可 以 获得 一 个 重要 的 推论 ,就 是 一 个 程序 可 以 一 半 放 在 磁盘 上 ,一半 放 在 内 存 
上 。 这 样 ,从 物理 上 讲 , 一 个 程序 发 出 的 访问 地 址 有 可 能 在 内 存 , 也 有 可 能 在 磁盘 。 

计算 机 怎么 能 知道 这 个 地 址 所 指向 的 数据 在 内 存 还 是 在 磁盘 上 呢 ? 这 无 法 在 静态 地 址 
编译 时 就 知道 。 唯 一 的 方法 就 是 动态 地 址 翻译 。 在 每 次 内 存 访 问 的 时 候 , 虚 拟 地 址 就 是 用 
户 每 次 看 到 的 地 址 ,这 个 地 址 只 是 一 个 抽象 , 它 需 要 由 内 存 管理 单元 进行 翻译 , 变 成 物理 内 
存 地 址 才能 使 用 。 由 于 这 个 翻译 是 在 程序 执行 过 程 中 发 生 , 因 此 称 为 动态 地 址 翻译 。 有 了 
动态 地 址 翻译 ,编译 器 和 用 户 进程 就 再 也 不 用 考虑 物理 地 址 了 。 

在 动态 地 址 翻译 环境 下 ,一 个 虚拟 地 址 仅 在 被 访问 的 时 候 才 需要 放 在 内 存 里 ,在 其 他 时 
候 并 不 需要 占用 内 存 。 由 于 动态 地 址 翻译 可 以 动态 地 改变 翻译 参数 或 过 程 ,因此 可 以 在 程 
序 加 载 到 不 同 的 物理 位 置 时 ,或 不 同 的 虚拟 地 址 占用 同一 物理 地 址 时 ,做 出 正确 翻译 。 在 使 
基 址 和 极限 管理 模式 下 ,不同 程序 进入 物理 内 存 时 ,只 需要 变更 基 址 和 极限 寄存 器 的 内 容 
即 可 。 


4.4.4 非 固定 分 区 的 内 存 管理 


前 面 说 过 固定 分 区 的 内 存 管理 优点 就 是 直截了当 。 最 大 的 分 区 就 是 能 容纳 你 这 个 程序 
的 分 区 。 其 缺点 : 一 是 程序 大 小 和 分 区 大 小 的 匹配 不 容易 令 人 满意 ; 二 是 很 僵硬 ,如 果 有 


内 存 管理 


媳 上 避 


握 作 系统 原理 


个 程序 比 最 大 的 分 区 大 怎么 办 呢 ? 三 是 地 址 空间 无 法 增长 ,如 果 程 序 在 运行 时 内 存 需 求 增 
长 怎么 办 ? 很 容易 想到 固定 分 区 为 什么 有 这 个 缺陷 ,因为 分 区 是 固定 大 小 。 这 样 ,我 们 自然 
想到 用 非 固定 分 区 的 方式 来 管理 多 道 编程 的 内 存 空间 。 

非 固定 分 区 的 思想 很 简单 : 除了 划分 给 操作 系统 的 空间 外 ,其 余 的 内 存 空 间 是 作为 一 
个 整体 存在 的 。 当 一 个 程序 需要 占用 内 存 空间 时 ,就 在 该 片 空间 里 面 分 出 一 个 大 小 刚刚 满 
足 程序 所 需 的 空间 ; 再 来 一 个 程序 , 则 在 剩 下 的 空间 里 面 再 这 样 分 出 一 块 来。 在 这 种 模式 
下 ,一 个 程序 可 以 加 载 到 任何 地 方 ,也 可 以 和 物理 内 存 一 样 大 。 例 如 ,一 开始 内 存 里 只 有 操 
作 系 统 。 这 时 候 进 程 A 来 了 ,我 们 从 最 底下 分 出 一 片 与 进程 A 大 小 一 样 的 内 存 空间 ; 然后 
进程 B 来 了 ,我 们 在 进程 A 上 面 的 大 片 空间 分 出 一 片 与 进程 B 大 小 一 样 的 内 存 空间 ; 然后 
进程 C 来 了 ,我 们 就 在 进程 B 上 面 分 出 一 片 与 C 大 小 一 样 的 内 存 空 间 。 这 样 ,进程 A、 进 程 
B、 进 程 C 的 起 始 地 址 都 不 是 固定 的 ,如 图 4-11 所 示 。 

这 种 非 固定 分 区 内 存 管 理 的 机 制 也 很 简单 ,就 是 在 4. 6. 3 节 中 讲 过 的 基 址 和 极限 。 对 
每 一 个 程序 配备 两 个 寄存 器 : 基 址 寄存 器 和 极限 寄存 器 。 所 有 访问 地 址 都 必须 在 这 两 个 寄 
存 器 值 框 定 的 空间 里 ,否则 就 算 非法 访问 。 

仔细 的 读者 可 能 已 经 看 出 , 非 固定 分 区 这 种 管理 方式 存在 一 个 重大 问题 ; 每 个 程序 像 
县 罗汉 一 样 累积 ,如 果 程 序 B 在 执行 过 程 中 需要 更 多 空间 ,怎么 办 ? 例如 ,一 个 程序 在 执行 
过 程 中 不 断 产 生 新 的 数据 从 而 造成 数据 所 需 空间 的 增长 ,而 不 断 进行 嵌 套 函数 调用 的 时 候 
又 会 造成 所 需 栈 空间 的 增长 。 解 决 的 办 法 当然 是 在 一 开始 给 程序 分 配 空间 时 就 分 配 足够 大 
的 空间 , 留 有 一 片 闲 置 空间 供 程序 增长 用 ,如 图 4-12 所 示 。 
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图 4-12 中 A、B 两 个 程序 分 到 的 空间 都 大 于 其 实际 使 用 的 空间 。 因 此 ,程序 的 扩展 得 
到 支持 。 只 要 程序 增长 不 超过 所 分 配 的 空间 ,程序 的 增长 将 不 受 霸 绊 。 

不 过 ,在 分 配 增长 空间 后 需要 考虑 一 个 问题 。 一 个 程序 的 空间 增长 通常 有 两 个 来 源 : 
数据 和 栈 。 如 何 处 理 这 两 种 空间 增长 的 关系 会 对 整个 程序 的 扩展 性 产生 影响 。 

最 简单 的 方式 是 数据 和 栈 往 一 个 方向 增长 。 在 这 种 模式 下 ,我们 事先 给 数据 部 分 和 栈 
部 分 分 别 留 下 增长 空间 。 这 样 的 优点 是 两 者 独立 性 高 ,缺点 是 空间 利用 率 可 能 较 低 。 例 如 ， 
如 果 数 据 在 下 , 栈 在 上 , 当 栈 长 到 程序 所 分 配 空间 的 项 时 ,就 无 法 再 长 了 。 即 使 下 面 的 数据 
部 分 还 有 闲置 空间 也 不 能 利用 。 反 之 ,如 果 数 据 部 分 长 到 了 栈 的 底 后 也 无 法 再 增长 ,即使 栈 


的 上 方 还 有 空间 。 这 种 情况 如 图 4-13 所 示 。 

当然 ,可 以 通过 移动 栈 底 来 解决 上 述 问 题 。 但 这 样 成 本 就 高 了 ,而 且 十 分 复杂 ,容易 出 
错 。 一 个 更 简单 的 办 法 是 让 数据 和 栈 往 相 反方 向 增长 。 这 样 , 只 要 本 程序 的 自由 空间 还 有 
多 余 , 不 仅 可 以 进行 函数 调用 ,又 可 以 增加 新 的 数据 ,可 以 最 大 限度 地 利用 这 片 自由 空间 。 
这 是 UNIX 采取 的 策略 ,如 图 4-14 所 示 。Windows 内 存 管 理 则 为 栈 限制 了 边界 ,超出 该 边 
界 将 造成 程序 错误 并 导致 程序 终止 。 


图 4-13 栈 和 数据 同 向 增长 图 4-14 栈 和 数据 逆向 增长 


不 过 ,这 里 还 有 个 问题 : 操作 系统 怎么 知道 应 该 分 配 多 少 空间 给 一 个 程序 呢 ? 怎么 知 
道 该 程序 会 进行 多 少 层 嵌 套 调用 ,产生 多 少 新 的 数据 呢 ? 如 果 为 保险 起 见 分 配 一 个 很 大 的 
空间 ,就 有 可 能 造成 浪费 ; 而 分 配 小 了 , 则 可 能 造成 程序 无 法 继续 运行 。 

那 还 有 没有 别 的 办 法 呢 ? 有 ,给 该 程序 换 一 个 空间 。 就 是 当 一 个 程序 所 占 空 间 不 够 时 ， 
将 其 倒 到 磁盘 上 ,再 加 载 到 一 片 更 大 的 内 存 空间 。 这 种 将 程序 倒 到 磁盘 上 ,再 加 载 进 内 存 的 
管理 方式 称 为 交换 (Swap) 。 


4.4.5 交换 


交换 就 是 将 一 个 进程 从 内 存 倒 到 磁盘 上 ,再 将 其 从 磁盘 上 加 载 到 内 存 中 的 过 程 。 这 种 
交换 的 主要 目的 是 使 程序 找到 一 片 更 大 的 空间 ,从 而 防止 一 个 程序 因 空 间 不 够 而 期 省。 交 
换 的 另 一 目的 ,是 实现 进程 切换 ,也 就 是 将 一 个 程序 暂停 一 会 儿 , 让 另 一 个 程序 运行 。 不 过 
使 用 交换 进行 进程 切换 的 成 本 颇 高 ,一 般 不 这 样 做 。 

例如 ,在 图 4-15(a) 的 状态 下 ,进程 A 因为 空间 大 小 而 无 法 继续 执行 ,我 们 便 将 进程 A 
交换 到 磁盘 上 ,等 待 大 片 空间 的 出 现 ( 见 图 4-15(b))。 这 时 进程 D 因为 占用 空间 小 ,可 以 执 
行 ,因此 把 进程 D 加 载 进 来 ( 见 图 4-15(c))。 进 程 B 执 行 结束 后 空 出 一 片 空间 ,和 本 来 的 剩 
余 空 间 合 并 后 形成 了 一 个 可 以 容纳 进程 A 的 空间 ( 见 图 415(d)), 这 时 将 进程 A 再 加 载 进 


操作 系统 操作 系统 操作 系统 


(a) (©) (d) (©) 


图 4-15 进程 A 因 空 间 增长 而 交换 到 更 大 的 空间 里 
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来 ,形成 图 4-15(e) 的 状态 。 

交换 和 非 固定 分 区 一 样 ,每 个 程序 占用 一 片 连续 的 空间 ,操作 系统 使 用 基 址 和 极限 来 进 
行 管理 。 但 由 于 一 个 程序 在 执行 中 可 能 发 生 交换 ,其 基 址 和 极限 均 有 可 能 发 生变 化 。 但 这 
种 变化 对 于 内 存 管理 来 说 ,并 不 增加 多 少 难度 。 只 要 每 次 加 载 程序 的 时 候 将 基 址 和 极限 寄 
存 器 的 内 容 进行 重 载 即 可 。 


4.4.6 重生 


前 面 说 过 ,如 果 一 个 程序 在 执行 过 程 中 占用 空间 增 大 了 ,可 以 通过 交换 给 它 找 一 个 更 大 
的 空间 来 执行 。 这 种 情况 下 该 程序 的 增长 无 法 超过 物理 内 存 空 间 的 容量 。 如 果 一 个 程序 超 
过 了 物理 内 存 , 还 能 运行 吗 ? 

答案 也 许 出 平 意 料 。 能 。 这 个 办 法 就 是 重 又 (Overlay)。 重 县 就 是 将 程序 按照 功能 分 
成 一 段 一 段 功能 相对 完整 的 单元 。 一 个 单元 执行 完 后 ,再 执行 下 一 个 单元 ,而 一 旦 执行 到 下 
一 个 单元 ,就 不 会 再 执行 前 面 的 单元 。 所 以 可 以 把 后 面 的 程序 单元 覆盖 到 当前 程序 单元 上 。 
这 样 就 可 以 执行 一 个 比 物理 内 存 还 要 大 的 程序 。 

但 是 这 相当 于 把 内 存 管理 的 部 分 功能 交 给 了 用 户 ,是 个 很 拙劣 的 方法 。 况 且 也 不 是 每 
个 人 都 能 够 将 程序 分 成 边界 清晰 的 一 个 个 执行 单元 的 。 而 且 , 从 根本 上 说 这 不 能 算是 操作 
系统 提供 的 解决 方案 。 


4.4.7 双 基 址 
如 果 运 行 两 个 一 样 的 程序 ,只 是 数据 不 同 ,我 们 自然 想到 能 否 让 两 个 程序 共享 部 分 内 存 


空间 。 例 如 ,如 果 同 时 启动 两 个 PPT 演示 文稿 ,我 们 希望 PPT 的 程序 代码 部 分 能 共享 。 但 
在 基 址 极限 这 种 管理 模式 下 ,这 种 共享 无 法 实现 ,如 图 4-16 所 示 。 
数据 (D2) | 、、、 
进程 1 SA 进程 2 
虚拟 由 虚拟 地 址 


图 4-16 使 用 基 址 极限 难以 实现 程序 共享 


那么 有 什么 办 法 共享 这 段 代 码 又 不 容易 出 错 呢 ? 答案 就 是 设 定 两 组 基 址 和 极限 。 数 据 
和 代码 分 别 用 一 组 基 址 和 极限 表示 ,这 就 解决 了 问题 。 


4.5 空闲 空间 管理 


在 管理 内 存 的 时 候 ,操作 系统 需要 知道 内 存 空 间 有 多 少 空闲 。 如 何 才能 知道 有 哪些 空 
闲 呢 ? 这 就 必须 跟踪 内 存 的 使 用 。 跟 踪 的 办 法 有 两 种 : 第 一 种 办 法 是 给 每 个 分 配 单元 赋予 


个 字 位 ,用 来 记录 该 分 配 单元 是 否 闲置 。 例 如 , 字 位 取 值 0 表示 分 配 单元 闲置 , 字 位 取 值 
1 表示 该 分 配 单元 已 被 占用 。 这 种 表示 法 就 是 所 谓 的 位 图 表示 法 ,如 图 4-17 所 示 。 
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图 4-17 内 存 分 配 位 图 


从 图 4-17 中 可 以 看 出 ,内 存 空 间 最 前 面 的 4 个 分 配 单元 已 经 被 占用 , 接 下 来 是 5 个 分 
配 单元 则 处 于 闲置 状态 ,可 以 供 程序 使 用 。 其 他 以 此 类 推 。 

位 图 表示 法 的 优点 是 直观 简单。 在 搜索 需要 的 闲置 空间 时 只 需要 找到 一 片 0 个 数 大 
于 等 于 所 需 分 配 单元 数 即 可 。 例 如 ,如 果 需 要 找 一 片 4 个 分 配 单元 大 的 闲置 空间 , 则 通过 扫 
描 位 图 表 , 找 到 4 个 0 即 可 。 将 这 片 空间 分 配给 需要 的 程序 ,并 将 相应 的 位 图 值 设置 为 1。 

另外 一 种 办 法 是 将 分 配 单元 按 是 否 闲置 链接 起 来 ,这 种 办 法 称 为 链表 表示 法 。 对 于 
图 4-17 的 位 图 所 表示 的 内 存 分 配 状态 ,如 果 用 链表 表示 则 如 图 4-18 所 示 。 
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图 4-18 内 存 分 配 的 链表 表示 


+ 


图 中 的 P 代表 程序 , 即 当前 这 片 空间 由 程序 占用 。 后 面 的 数字 是 本 片 空间 的 起 始 分 配 
单元 号 和 大 小 。H 代表 的 是 空洞 , 即 这 是 一 片 闲置 空间 。 例 如 ,图 中 的 第 一 个 链表 项 表示 
一 片 大 小 为 4 个 分 配 单元 的 程序 ,起 始 地 址 为 第 0 个 分 配 块 。 第 二 个 链表 项 表示 一 片 大 小 
为 5 个 分 配 单元 的 闲置 空间 块 ,起 始 地 址 为 第 4 个 分 配 单元 。 其 他 以 此 类 推 。 

在 链表 表示 下 ,寻找 一 个 给 定 大 小 的 闲置 空间 意味 着 找到 一 个 类 型 为 H 的 链表 项 ， 
其 大 小 大 于 或 等 于 给 定 的 目标 值 。 不 过 ,扫描 链表 速度 通常 较 慢 。 为 提高 查找 闲置 单元 
的 速度 ,有 人 提出 了 将 闲置 空间 和 被 占 空 间 分 开设 置 链表 ,这 样 就 形成 了 两 个 链表 的 管 
理 模式 。 

位 图 表示 和 链表 表示 各 有 优 缺 点 。 如 果 程 序数 量 很 少 ,那么 链表 比较 好 ,因为 链表 的 表 
项 数量 少 。 例 如 ,如 果 只 有 三 个 程序 在 内 存 中 , 则 最 多 只 需要 7 个 链表 结 点 。 但 是 如 果 程 序 
很 稠密 ,那么 链表 的 结 点 就 很 多 了 。 

位 图 表示 法 的 空间 成 本 是 固定 的 , 它 不 依赖 于 内 存 中 程序 的 数量 。 因 此 ,从 空间 成 本 上 
分 析 , 到 底 使 用 哪 种 表示 法 要 看 链表 表示 后 的 空间 成 本 是 大 于 位 图 表示 还 是 小 于 位 图 表示 
而 定 。 

从 可 靠 性 上 看 ,位 图 表示 法 没有 容错 能 力 。 如 果 一 个 分 配 单 元 为 1, 并 不 能 肯定 它 应 该 
为 1, 还 是 因为 错误 变 成 1 的 ,因为 链表 有 被 占 空间 和 闲置 空间 的 表 项 ,可 以 相互 验证 ,具有 
一 定 的 容错 能 力 。 

从 时 间 成 本 上 ,位 图 表示 法 在 修改 分 配 单元 状态 时 ,操作 很 简单 ,直接 修改 其 位 图 值 即 
可 ,而 链表 表示 法 则 需要 对 前 后 空间 进行 检查 以 便 做 出 相应 的 合并 。 例 如 ,在 图 4-18 所 示 的 
情况 下 ,如 果 程 序 中 间 的 那个 程序 (占用 位 置 从 11 开始 ,长 度 为 和) 终止 , 则 链表 将 如 图 4-19 所 
示 。 如 果 是 最 前 面 的 程序 终止 , 则 链表 将 如 图 4-20 所 示 。 
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图 4-19 中 间 程序 空间 释放 时 链表 项 的 合并 
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图 4-20 第 一 个 程序 空间 释放 时 链表 项 的 合并 


a 


当然 ,还 可 以 从 查找 时 间 上 进行 分 析 。 这 个 留 给 读者 作为 练习 。 
小 结 


存储 器 是 计算 机 系统 的 重要 组 成 部 分 。 存 储 管理 对 主 存 中 的 用 户 区 进行 管理 ,其 目的 
是 尽 可 能 地 提高 主 存 空间 的 利用 率 , 使 主 存在 成 本 、 速 度 和 规模 之 间 获 得 较 好 的 平衡 。 存 储 
管理 的 基本 功能 有 : 主 存 空 间 的 分 配 与 回收 ,地址 转换 、 主 存 空间 的 共享 与 保护 、 主 存 空间 
的 扩充 。 

在 多 道 程序 设计 系统 中 ,为 了 方便 程序 编制 ,用 户 程序 中 使 用 的 地 址 是 逻辑 地 址 ,而 
CPU 则 是 按 物 理 地 址 访问 主 存 、 读 取 指令 和 数据 。 为 了 保证 程序 的 正确 执行 ,需要 进行 地 
址 转换 。 地 址 转换 又 称 为 重 定位 ,有 静态 重 定位 和 动态 重 定位 。 采 用 动态 重 定位 的 系统 支 
持 程 序 的 浮动 。 

早期 单 用 户 单 任务 操作 系统 中 主 存 管理 采用 单 用 户 连续 存储 管理 方式 。 现 代 操 作 系统 
支持 多 道 程 序 设计 ,满足 多 道 程序 设计 最 简单 的 存储 管理 技术 是 分 区 管理 ,有 固定 分 区 管理 
和 可 变 分 区 管理 。 分 区 管理 中 ,当主 存 空 间 不 足 时 ,交换 技术 和 覆盖 技术 可 以 达到 扩充 主 存 
的 目的 。 


第 5 章 页 式 和 段 式 内 存 管 理 


分 区 存储 管理 方式 尽管 实现 较为 简单 ,但 要 求 把 一 个 进程 放置 在 一 段 连续 的 内 存 区 域 
中 ,从 而 造成 严重 的 碎片 问题 , 导致 内 存 的 利用 率 较 低 。 为 了 解决 分 区 存储 管理 存在 的 问 
题 , 人 们 提出 了 页 式 和 段 式 内存 管 理 方式 。 


5.0 问题 导入 


当 把 程序 从 外 存 装 入 到 内 存 中 创建 进程 时 ,需要 考虑 内 存 是 否 能 够 容纳 该 进程 ? 在 多 
任务 环境 下 ,应 该 把 程序 装 入 哪 一 个 可 用 的 存储 区 域 ? 哪些 内 存 区 域 是 空闲 的 ? 哪些 区 域 
是 已 经 被 占用 的 ?如 何 记 住 内 存 占用 情况 ?如 果 内 存 不 足以 容纳 程序 怎么 办 ? 程序 在 运行 
过 程 中 如 果 要 动态 申请 内 存 空 间 应 如 何 处 理 ? 一 个 进程 如 果 要 访问 另 一 个 进程 或 操作 系统 
的 存储 区 域 ,应 如 何 处 理 ? 


5.1 页 式 内 存 管 理 


页 式 内 存 管理 技术 允许 进程 的 物理 地 址 空间 是 连续 的 ,这 样 ,可 以 把 一 个 程序 分 散在 各 
个 空闲 的 物理 块 中 , 它 既 不 需要 移动 内 存 中 原 有 的 信息 ,又 解决 了 外 部 碎片 的 问题 ,从 而 提 
高 了 内 存 的 利用 率 , 因 此 ,页 式 内 存 管 理 技术 通常 被 绝 大 多 数 操作 系统 所 采用 。 


5.1.1 基本 原理 


在 页 式 存储 管理 方式 中 ,把 用 户 程序 的 地 址 空间 划分 为 若干 个 大 小 相等 的 区 域 ,每 个 
域 称 为 页 面 或 页 。 每 个 页 面 都 有 一 个 编号 , 称 为 页 号 。 页 号 一 般 从 0 开始 编号 ,如 0,1， 
es 把 内 存 空 间 划 分 成 若干 和 页 面 大 小 相等 的 物理 块 ,这 些 物理 块 称 为 内 存 块 。 同 样 ,每 
个 物理 块 也 有 一 个 编号 ,物理 块 的 编号 从 0 开始 依次 排列 。 

在 页 式 内 存 管理 系统 中 ,页 面 的 大 小 是 由 硬件 的 地 址 结构 所 决定 的 。 只 要 机 器 确定 了 ， 
页 面 大 小 即 可 确定 。 一 般 来 说 ,页 面 的 大 小 选择 为 2 的 若干 次 窘 ,根据 地 址 结构 的 不 同 ,其 
大 小 从 512B 到 16MB 不 等 。 

用 户 程 序 开始 执行 后 ,会 将 其 逻辑 地 址 转换 成 物理 地 址 。 当 程序 的 地 址 空间 小 于 主 存 
可 用 空间 时 ,只 要 求 把 当前 需要 的 一 部 分 页 面 装 入 主 存 即 可 ,这 样 对 虚 地 址 空间 的 限制 就 被 
取消 了 。 

页 式 存储 管理 系统 需要 解决 的 几 个 问题 是 : 

(1) 地 址 映射 


x 


所 作 系 统 原 理 


(2) 页 面 置 换 策略 。 
5.1.2 分 页 内 存 管理 


1. 页 表 

在 分 页 内 存 管理 系统 中 ,由 CPU 生成 的 每 个 地 址 被 硬件 分 成 两 个 部 分 : 页 号 (p) 和 页 
内 偏 移 量 (xe) 。 通 常 ,如 果 逻 辑 地 址 空间 为 2 , 且 页 面 的 大 小 为 2" 单元 ,那么 ,逻辑 地 址 的 
高 m 一 n 位 表示 页 号 ,而 低 n 位 表示 页 内 偏 移 量 。 这 样 ,一 个 地 址 长 度 为 20 位 的 计算 机 系 
统 ,如 果 每 页 的 大 小 为 1KB(2”), 则 其 地 址 结构 如 图 5-1 所 示 。 
19 10 9 0 在 分 页 内 存 管 理 方式 中 ,系统 以 物理 块 为 单位 把 内 存 
页 号 p 页 内 偏 移 w 分 配给 各 个 进程 ,进程 的 每 个 页 面 对 应 一 个 内 存 的 物理 块 ， 
图 5-1 页 式 系统 的 地 址 结构 ” 并且, 一 个 进程 的 若干 个 页 面 可 以 分 别 装 入 物理 上 不 连续 
的 内 存 物理 块 ,如 图 5-2 所 示 。 当 把 一 个 进程 装 入 内 存 时 ， 
首先 检查 它 有 多 少 页 。 如 果 它 有 nn 页 , 则 内 存 里 至 少 应 该 有 nn 个 空闲 的 物理 块 才能 装 入 该 
进程 。 如 果 满足 要 求 , 则 分 配 个 空闲 的 物理 块 给 该 进程 ,并 将 其 装 入 , 且 在 该 进程 的 页 表 
中 记录 各 个 页 面 对 应 的 内 存 物理 块 的 块 号 。 从 图 5-2 可 以 看 出 ,进程 1 的 页 面 是 连续 的 ,而 
装 入 内 存 后 ,内 存 分 配给 进程 1 的 物理 块 是 不 相 邻 的 ,例如 ,0 共 页 放 在 3# 物理 块 ,1# 页 放 

在 5# 物 理 块 ,等 等 。 


块 号 
0 
1 
第 0 页 . 
i 进 穆 ! 第 0 页 | 3 
3 作 
上 全 -一 | 4 
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第 4 页 进程 ! 第 2 页 6 
进程 1 第 -1 页 | 7 
a 进程 第 4 页 | 8 
第 n 一 1 页 
y 3 9 
pl 进程 第 页 | 10 
m-2 
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内 存 


图 5-2 分 页 内 存 管理 系统 


在 分 页 内 存 管理 系统 中 ,允许 将 进程 的 各 页 高 散 地 装 入 内 存 的 任何 空闲 物理 块 中 ,这 样 
就 出 现 了 进程 页 号 连续 ,而 物理 块 号 不 连续 的 情况 。 为 了 找到 每 个 页 面 在 内 存 中 对 应 的 物 
理 块 ,系统 为 每 个 进程 建立 了 一 张 页 面 映射 表 , 简 称 页 表 。 进 程 的 所 有 页 面 依次 在 页 表 中 有 
一 个 页 表 项 ,记载 着 相应 页 面 在 内 存 中 对 应 的 物理 块 号 。 当 进程 执行 时 ,按照 逻辑 地 址 中 的 
页 号 在 页 表 中 查找 对 应 的 页 表 项 ,找到 该 页 号 在 内 存 中 对 应 的 物理 块 号 。 页 表 的 作用 就 是 
实现 页 号 到 物理 块 号 的 地 址 映射 , 即 逻 辑 地 址 到 物理 地 址 的 映射 。 


2. 地 址 结构 
对 于 某 台 具体 的 机 器 而 言 ,其 地 址 结构 是 一 定 的 。 如 果 给 定 的 逻辑 地 址 是 A, 页 面 的 大 
小 是 工 , 则 页 号 p 和 页 内 偏 移 w 可 按照 下 列 公式 计算 : 
B= INT[ 全 |] w = AMODL 
其 中 ,INT 是 向 下 整除 的 函数 ,MOD 是 取 余 数 的 函数 。 例 如 , 设 系统 页 面 大 小 是 1KB， 
A=2354, 则 ; 


新 二 INT [434 |= 2 w= 2354 MOD 1024 = 306 


3. 地 址 映射 

在 分 页 内 存 管理 系统 中 ,利用 页 表 来 实现 用 户 程 序 的 逻辑 地 址 和 实际 物理 地 址 的 转换 。 
每 个 作业 有 一 个 页 表 , 页 表 通 常 存放 在 内 存 中 。 在 系统 中 设置 一 个 页 表 寄 存 器 ,在 其 中 存放 
页 表 在 内 存 中 的 起 始 地 址 和 页 表 长 度 。 作 业 未 执行 时 ,页 表 的 起 始 地 址 和 页 表 长 度 存放 在 
本 作业 的 JCB 中 。 当 调度 程序 调度 到 某 作 业 时 , 才 将 这 两 个 数据 装 入 页 表 寄 存 器 。 当 作业 
要 访问 某 个 逻辑 地 址 中 的 指令 或 数据 时 ,分 页 内 存 管 理 系统 的 地 址 变换 机 构 自 动 将 有 效 地 
址 分 为 页 号 和 页 内 地 址 ( 即 页 内 偏 移 量 ) 两 个 部 分 ,再 以 页 号 为 索引 检索 页 表 。 整 个 查找 过 
程 由 硬件 执行 。 

在 执行 检索 之 前 , 先 将 页 号 和 页 表 长 度 进行 比较 ,如 果 页 号 大 于 或 等 于 页 表 长 度 , 则 表 
示 本 次 访问 的 地 址 已 超出 作业 的 地 址 空间 。 因 此 ,系统 将 捕获 这 个 错误 并 产生 一 个 地 址 越 
界 中 断 。 如 果 没 有 出 现 地 址 越界 错误 , 则 从 页 表 中 得 到 该 页 号 对 应 的 物理 块 号 ,把 它 装 入 物 
理 地 址 寄存 器 。 同 时 ,将 页 内 地 址 直接 送 入 物理 寄存 器 的 块 内 地 址 字段 中 。 这 样 ,物理 地 址 
寄存 器 中 的 内 容 就 是 由 物理 块 号 和 块 内 地 址 拼接 而 成 的 实际 物理 地 址 ,从 而 完成 从 逻辑 地 
址 到 物理 地 址 的 映射 。 

图 5-3 描述 了 作业 1 程序 中 的 一 条 指令 的 执行 情况 ,用 以 说 明 分 页 内 存 管理 系统 的 地 
址 映射 过 程 。 程 序 地 址 空间 的 第 200 号 单元 处 有 一 条 指令 为 "mov rl,[2052]”。 这 条 指令 
在 主 存 中 的 实际 位 置 为 2248 号 单元 ,而 操作 数 12 345 的 逻辑 地 址 为 2052 号 单元 , 它 的 物 
理 地 址 是 7172 号 单元 。 
当 作业 1 的 相应 进程 在 CPU 上 运行 时 ,操作 系统 负责 把 该 作业 的 页 表 在 主 存 中 的 起 始 
地 址 (a) 送 到 页 表 地 址 寄存 器 中 ,以 便 在 进程 运行 过 程 中 进行 地 址 映射 时 能 快速 地 找到 该 
作业 的 页 表 。 当 作业 1 的 程序 执行 到 指令 “mov rl,[2052]” 时 ,CPU 给 出 的 操作 数 地 址 为 
2052, 首 先 由 分 页 机 构 自 动 地 把 它 分 为 两 部 分 ,得 到 页 号 p 二 2, 页 内 偏 移 ww 一 4。 然 后 ,根据 
页 表 地 址 寄存 器 指示 的 页 表 起 始 地 址 ,以 页 号 为 索引 ,找到 第 二 页 所 对 应 的 物理 块 号 为 7。 
最 后 ,将 物理 块 号 7 和 页 内 偏 移 量 4 拼接 在 一 起 :就 形成 了 访问 主 存 的 物理 地 址 7172。 这 
正 是 所 取 数 据 12 345 所 在 主 存 的 实际 位 置 。 

由 上 述 地 址 映射 过 程 可 知 , 在 分 页 内 存 管 理 系统 环境 下 ,程序 员 编 制 的 程序 ,或 由 编译 
程序 给 出 的 目标 程序 ,经 装配 链接 后 形成 一 个 连续 的 地 址 空间 ,其 地 址 空间 的 分 页 由 系统 自 
动 完 成 ,而 地 址 映射 则 通过 页 表 自 动 地 、 连 续 地 进行 ,系统 的 这 些 功能 对 用 户 或 程序 员 而 言 
是 透明 的 。 正 因为 在 分 页 系统 中 ,地 址 映射 过 程 主要 是 通过 页 表 来 实现 的 ,因此 ,人 们 称 页 
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: 
作业 1 页 表 
图 5-3 分 页 内 存 管理 系统 的 地 址 映射 过 程 


表 为 地 址 变换 表 或 地 址 映射 表 。 


5.1.3 分 页 系统 的 优 缺点 


采用 分 页 内 存 管理 技术 不 会 产生 外 部 碎片 ,但 是 可 能 产生 内 部 碎片 。 由 于 分 页 内 存 管 
理 系 统 的 内 存 分 配 是 以 物理 块 为 单位 进行 的 ,如 果 进 程 要 求 的 内 存 不 是 页 大 小 的 整数 倍 , 那 
么 ,最 后 一 个 物理 块 就 用 不 完 , 从 而 导致 页 内 碎片 的 出 现 。 

分 页 系统 的 另 一 个 优点 是 可 以 共享 共同 的 代码 ,这 一 点 对 分 时 系统 特别 重要 。 


5.1.4 快 表 


为 了 提高 从 逻辑 地 址 向 物理 地 址 转换 过 程 中 地 址 的 变换 速度 ,可 在 地 址 变换 机 构 中 增 
设 一 个 具有 并 行 查询 能 力 的 特殊 高 速 缓冲 存储 器 ,又 称 为 “联想 寄存 器 ”(Associative 
Memory) 或 称 为 “ 快 表 ”。 在 IBM 系统 中 称 为 TLB(Translation Lookaside Buffer) ,存放 当 
前 访问 的 那些 页 表 项 。 

具有 快 表 的 地 址 变换 步骤 如 下 。 

(1) 在 CPU 给 出 有 效 地 址 后 ,地 址 变换 机 构 自 动 将 页 号 p 送 入 高 速 缓冲 寄存 器 中 ,并 
将 此 页 号 与 高 速 缓存 中 的 所 有 页 号 进行 比较 。 

(2) 如 果 其 中 有 与 此 页 号 匹配 的 , 便 表示 所 要 访问 的 页 表 项 在 快 表 中 。 

(3) 直接 从 快 表 中 读 出 该 页 号 所 对 应 的 物理 块 号 ,并 送 到 物理 地 址 寄存 器 中 。 

(4) 如 果 在 快 表 中 未 找到 相同 的 页 表 号 , 则 必须 再 访问 内 存 中 的 页 表 , 从 页 表 中 找到 该 
页 号 所 对 应 的 页 表 项 后 ,把 从 页 表 项 中 读 出 的 物理 块 号 送 入 地 址 寄存 器 。 

(5) 同时 ,将 此 页 号 所 对 应 的 页 表 项 存 入 快 表 中 , 即 重新 修改 快 表 。 

具有 快 表 的 地 址 变换 过 程 , 如 图 5-4 所 示 。 
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图 5-4 具有 快 表 的 地 址 变换 过 程 


5.1.5 页 共享 与 保护 


1， 页 共享 

分 页 系统 可 以 共享 共同 的 代码 ,这 对 分 时 系统 特别 重要 。 设 想 一 个 系统 : 该 系统 有 40 
个 用 户 , 每 个 用 户 都 执行 一 个 文本 编辑 器 。 文 本 编辑 器 有 150KB 代码 段 和 50KB 数据 段 ， 
需要 8000KB 来 支持 40 个 用 户 。 然 而 ,如 果 代 码 是 可 重 入 代码 , 则 可 以 共享 ,如 图 5-5 
所 示 。 
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进程 B 。”B 的 页 表 
图 5-5 在 分 页 系统 中 的 代码 共享 


可 重 入 代码 (或 纯 代码 ) 是 在 其 执行 过 程 中 本 身 不 做 任何 修改 的 代码 ,通常 由 指令 和 常 
数组 成 。 编 辑 器 有 三 页 ,每 页 的 大 小 为 50KB, 这 些 页 只 是 为 了 说 明 问题 而 已 ,可 为 三 个 进 
程 共享 。 每 个 进程 都 有 自己 的 数据 页 。 

此 时 ,只 需要 在 物理 内 存 中 保存 一 个 编辑 器 副本 。 每 个 用 户 的 页 表 映 射 到 编辑 器 的 同 
一 物理 副本 ,而 数据 页 映射 到 不 同 的 物理 块 。 因 此 ,为 了 支持 40 个 用 户 , 只 需要 一 个 编辑 副 
本 (150KB) ,再 加 上 40 个 用 户 空间 50KB, 总 的 需求 空间 为 2150KB, 而 不 是 8000KB, 这 是 
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一 个 重要 的 节省 。 

2. 存储 保护 

现代 操作 系统 中 主 存储 器 由 多 个 用 户 程序 所 共享 。 为 了 保证 多 个 应 用 程序 之 间 互 不 影 
响 , 必 须 由 硬件 .软件 配 合 保证 每 个 程序 只 能 在 给 定 的 存储 区 域 活动 ,这 种 措施 称 为 存储 保 
护 。 存 储 保护 在 采用 分 页 存储 管理 技术 的 系统 中 容易 实现 , 且 十 分 有 效 。 因 为 ,被 保护 的 是 
一 个 程序 。 存 储 保护 的 目的 是 防止 用 户 程 序 之 间 相 互 干扰 。 而 在 分 页 系统 中 ,由 于 页 的 划 
分 是 物理 的 分 割 ,没有 逻辑 含义 ,所 以 存储 保护 并 不 十 分 有 效 。 


5.1.6 内 存 抖动 


1. 抖动 产生 的 原因 

当主 存 空间 已 经 装 满 ,而 又 需要 转 入 新 的 页 面 时 ,必须 按照 一 定 的 算法 把 已 经 在 内 存 中 
的 一 些 页 面 调 出 ,这 个 工作 称 为 页 面 蔡 换 。 因 此 ,页面 更 新 算法 就 是 用 来 确定 应 该 淘汰 哪些 
页 面 的 算法 ,也 称 为 淘汰 算法 。 

算法 的 选择 很 重要 ,如 果 选 用 了 一 个 不 合适 的 算法 ,就 会 出 现 这 样 的 现象 : 刚刚 被 淘汰 
的 页 面 又 要 立即 使 用 ,因此 又 要 把 它 重新 调 入 内 存 , 而 调 入 不 久 后 又 被 淘汰 , 调 出 内 存 ,淘汰 
不 久 后 又 被 调 入 内 存 。 如 此 反复 ,使 得 整个 系统 的 页 面 调度 非常 的 频繁 ,以 至 于 大 部 分 时 间 
都 花费 在 反复 调度 页 面 上 。 这 种 现象 称 为 “抖动 ,又 称 为 “ 颠 佬 ”, 一 个 好 的 调度 算法 应 尽 可 
能 地 减少 和 避免 内 存 拌 动 现象 。 

2. 防止 抖动 的 方法 

防止 抖动 发 生 或 者 限制 抖动 影响 有 多 种 方法 。 由 于 抖动 产生 的 原因 ,这 些 方 法 都 是 基 
于 调节 多 道 程 序 的 度 。 

1) 采用 局 部 置换 策略 

如 果 一 个 进程 出 现 拌 动 , 它 不 能 从 另外 的 进程 那里 获取 内 存 块 , 不 会 引发 其 他 进程 出 现 
抖动 ,使 抖动 局 限于 一 个 小 范围 内 。 然 而 ,这 种 方法 并 未 消除 抖动 的 发 生 , 而 且 在 一 些 进程 
发 生 拌 动 的 情况 下 ,等 待 磁盘 1/O 的 进程 增多 ,使 得 平均 缺 页 处 理 时 间 加 长 ,延长 了 有 效 存 
取 时 间 。 

2) 挂 起 某 些 进程 

当 出 现 CPU 利用 率 很 低 而 磁盘 I/O 非常 频繁 的 情况 时 ,可 能 因为 多 道 程 序 度 太 高 而 
造成 抖动 。 为 此 ,可 以 挂 起 一 个 或 几 个 进程 , 腾 出 内 存 空间 供 抖动 进 程 使 用 ,从 而 消除 抖动 
现象 。 被 挂 起 进程 的 选择 策略 有 多 种 ,如 选择 优先 权 最 低 的 进程 、 缺 页 进程 .最 近 激 活 的 进 
程 、 驻 留 集 最 小 的 进程 .最 大 的 进程 。 

3) 采用 缺 页 频 度 法 

抖动 发 生 时 , 缺 页 率 必 然 很 高 ,因此 可 以 通过 控制 缺 页 率 来 预防 抖动 。 如 果 缺 页 率 太 
高 ,表明 进程 需要 更 多 的 内 存 物理 块 ; 如 果 缺 页 率 很 低 , 表 明 进 程 可 能 占用 了 太 多 的 内 存 物 
理 块 。 这 里 规定 一 个 缺 页 率 , 依 次 设置 相应 的 上 限 和 下 限 。 如 果实 际 缺 页 率 超 出 上 限 值 , 就 
为 该 进程 分 配 另 外 的 内 存 物理 块 ; 如 果实 际 缺 页 率 低 于 下 限 值 ,就 从 该 进程 的 驻 留 集中 取 
走 一 个 内 存 物理 块 。 通 过 直接 测量 和 控制 缺 页 率 , 可 以 避免 抖动 。 


5.2 页 面 更 新 算法 


当 程 序 运行 中 需要 调 入 新 的 页 面 而 先前 分 得 的 主 存 物理 块 已 经 用 完 的 时 候 , 需 要 淘汰 
一 页 。 系 统 应 提供 淘汰 机 制 和 淘汰 策略 ,包括 扩充 页 表 数 据 项 、 确 定 页 面 淘 汰 原则 和 是 否 需 
要 淘汰 页 面 的 判断 及 处 理 。 本 节 主 要 介绍 三 种 常用 的 页 面 更 新 算法 。 


5.2.1 页 面 交 换 机 制 


页 面 交换 ,又 称 为 页 面 置 换 , 页 面 更 新 或 页 面 淘 汰 。 当 请 求 调 页 程序 需要 调 进 一 个 页 
面 , 而 此 时 该 作业 所 分 得 的 主 存 物理 块 已 经 全 部 用 完 , 则 必须 淘汰 该 作业 已 经 在 主 存 中 的 一 
个 页 面 。 

为 了 给 置换 页 面 提 供 依据 ,页 表 中 还 必须 包含 关于 页 面 使 用 情况 的 信息 ,并 增设 专门 的 
硬件 和 软件 来 考查 和 更 新 这 些 信 息 。 这 说 明 页 表 的 功能 还 需要 进一步 的 扩充 。 于 是 ,在 页 
表 中 增加 了 引用 位 和 改变 位 。 

引用 位 是 用 来 指示 某 页 最 近 是 否 被 访问 :“0” 表 示 没 有 被 访问 过 ;“1” 表 示 已 经 被 访问 
过 。 改 变 位 是 表示 某 页 是 否 被 修改 :“1” 表 示 已 经 被 修改 过 ;“0” 表 示 未 被 修改 过 。 这 一 信 
息 是 为 了 在 淘汰 一 页 时 决定 是 否 需 要 写 回 辅 存 而 设置 的 。 因 此 ,这 种 情况 下 完整 的 页 表 结 
构 通常 在 逻辑 上 至 少 应 包含 如 图 5-6 所 示 的 各 个 数据 项 。 


页 号 物理 块 号 中 断 位 改变 位 引用 位 外 存 地 址 


图 5-6 完整 的 页 表 结 构 


页 式 系统 的 虚拟 存储 功能 是 由 硬件 和 软件 相配 合 实 现 的 ,该 过 程 也 说 明了 缺 页 处 理 和 
淘汰 页 面 的 处 理 。 如 图 5-7 所 示 的 是 指令 执行 步骤 和 缺 页 中 断 处 理 过 程 。 其 中 ,虚线 上 面 
部 分 是 由 硬件 实现 的 ,而 下 面部 分 通常 由 软件 实现 。 当 中 断 位 为 1 时 发 生 缺 页 中 断 , 由 调 页 
程序 得 到 控制 权 。 若 请 求 新 页 的 程序 还 有 空闲 块 , 即 可 直接 调 入 ; 否则 , 转 入 页 面 淘汰 子 程 
序 , 确 定 应 淘汰 的 页 面 。 图 5-7 给 出 的 只 是 粗略 的 框图 ,具体 过 程 相当 的 复杂 。 因 为 作业 程 
序 是 以 文件 形式 存 于 外 存 中 的 , 当 需 要 从 外 存 调 入 一 页 或 需要 重新 写 回 外 存 时 ,必然 会 涉及 
文件 系统 和 调用 输入 /输出 过 程 。 

在 多 进程 环境 下 ,一 个 进程 在 等 待 传输 页 面 时 , 它 处 于 阻塞 状态 ,此 时 ,系统 可 以 调度 另 
一 个 进程 运行 。 当 页 面 传输 完成 后 ,唤醒 原先 被 阻塞 的 那个 进程 ,等 到 下 次 再 调度 到 它 时 ， 
才能 恢复 到 原来 的 断 点 继续 运行 下 去 。 


5.2.2 最 优 更 新 算法 


最 优 (Optimal) 更 新 算法 是 由 Belady 于 1966 年 提出 的 一 种 理论 上 的 页 面 更 新 算法 。 
该 算法 所 选择 的 被 淘汰 页 面 ,将 是 以 后 永久 不 被 访问 ,或 者 是 在 未 来 最 长 时 间 内 不 再 被 访问 
的 页 面 。 采 用 最 优 更 新 算法 通常 可 以 保证 获得 最 低 的 缺 页 率 。 
假设 系统 为 某 个 进程 分 配 了 4 个 物理 块 , 并 考虑 有 以 下 页 面 号 引用 串 : 4、3、2、1、4、3、 
5.4:3.2,1,5。 


页 式 和 段 式 内 存 管理 


攻 串 避 


握 作 系统 原理 


一 | 启动 要 处 理 的 指令 |= 一 一 一 一 
生生 一 让 
计算 有 效 地 址 
1 取 下 一 条 指令 指 
取出 页 号 T 人 
硬 访问 内 存 ， 完 成 该 指令 入 
件 村 
| Hi Oe A 2Z-—--—-—-—-—-— 
软 
件 
地址 读 久 调整 页 表 及 
按 外 存 地 址 读 入 和 由 家 及 
所 需 的 页 面 存储 分 块 表 
调整 页 表 及 把 该 页 
存储 分 块 表 人 
=| 重新 启动 
| 被 中 断 的 指令 


图 5-7 ”指令 执行 步骤 和 缺 页 中 断 处 理 过 程 


运行 时 , 先 将 4、3、2、1 这 4 个 页 面 装 入 内 存 。 当 进程 要 访问 5# 页 面 时 ,将 会 产生 缺 页 
中 断 , 此 时 ,操作 系统 根据 最 优 更 新 算法 , 先 找 有 没有 未 来 永 不 被 访问 的 页 面 , 没 有 此 类 型 页 
面 , 则 再 寻找 未 来 最 长 时 间 不 再 被 访问 的 页 面 ,发 现在 当前 4 个 物理 块 中 的 页 面 4、3、2、1 
中 ,1# 页 是 最 晚 被 访问 的 ,因此 符合 要 求 , 则 选择 淘汰 1# 页 ,将 5# 页 放 入 1# 页 原来 所 占 
的 物理 块 内 。 如 果 此 时 找到 多 个 符合 要 求 的 页 面 , 则 从 中 任意 选择 一 个 页 面 淘汰 即 可 。 
表 5-1 给 出 了 采用 最 优 更 新 算法 时 页 面 的 置换 情况 。 
表 5-1 最 优 更 新 算法 页 面 置换 表 


页 面 走向 4 3 | 4 3 [3 4 3 2 1 5 
物理 块 0# 4 4 4 4 4 4 4 4 4 4 1 1 
物理 块 1# 3 3 3 3 3 3 3 3 是 3 3 
物理 块 2# 2 2 2 2 2 2 入 名 2 和] 
物理 块 3# 1 1 1 1 1 1 1 5 5 
是 否 缺 页 Y Eg Y 区 和 Y 


从 表 5-1 中 可 以 看 出 ,系统 总 共 发 生 了 6 次 页 面 置换 , 缺 页 率 为 6/12 一 0.5, 即 50%。 


5.2.3 先进 先 出 更 新 算法 


先进 先 出 (FIFO) 更 新 算法 是 最 早出 现 的 页 面 更 新 算法 。 该 算法 总 是 淘汰 最 先进 入 内 
存 的 页 面 , 即 选择 在 内 存 中 停留 时 间 最 长 的 一 页 予以 淘汰 。 如 果 同 时 有 多 个 页 面 符合 淘汰 
的 条 件 , 则 任意 选择 一 个 予以 淘汰 即 可 。 仍 然 以 上 面 的 例子 为 例 , 表 5-2 给 出 了 采用 先进 先 
出 更 新 算法 的 页 面 置换 过 程 。 


表 5-2 ”先进 先 出 更 新 算法 页 面 置换 表 


页 面 走向 4 3 2 1 4 3 5 4 3 2 1 5 
物理 块 0# 4 4 4 4 4 4 5 5 5 5 1 1 
物理 块 1# 3 3 3 3 3 3 4 4 4 4 5 
物理 块 2# 2 2 2 2 2 3 3 3 入 
物理 块 3# 1 1 1 1 1 1 肥 2 器 
是 否 缺 页 Y 总 党 你 实 Y Y YY 区 


当 进 程 第 一 次 访问 5# 页 时 ,将 把 在 内 存 中 停留 时 间 最 长 的 4# 页 置换 出 内 存 。 当 第 三 
次 访问 4# 页 时 , 则 把 当时 内 存 中 停留 最 长 的 3# 页 置换 出 。 从 表 5-2 中 可 以 看 出 ,系统 总 
共 发 生 了 10 次 页 面 置换 , 缺 页 率 为 10/12。 

先进 先 出 更 新 算法 的 优点 是 容易 理解 , 且 方 便 程 序 设计 。 然 而 , 它 的 性 能 并 不 是 很 好 。 
仅 当 按 线性 顺序 访问 地 址 空间 时 ,这 种 算法 才 是 最 理想 的 ,和 否则 ,该 算法 效率 不 高 ,因为 那些 
经 常 被 访问 的 页 面 通常 在 内 存 中 停留 的 时 间 最 长 ,而 它们 却 因为 变 “ 老 ”而 不 得 不 被 淘汰 
出 去 。 


5.2.4 最 近 最 久未 使 用 更 新 算法 


最 近 最 久未 使 用 (Least Recently Used,LRU) 更 新 算法 以 “最 近 的 过 去 ”作为 “不 久 的 将 
来 "的 近似 ,选择 最 近 一 段 时 间 内 最 久 没 有 使 用 的 页 面 淘 汰 。 它 的 实质 是 : 当 需 要 更 新 一 页 
时 ,选择 在 最 近 一 段 时 间 内 最 久 没 有 被 使 用 的 页 面 予 以 淘汰 。 仍 然 以 上 面 的 例子 为 例 ， 
表 5-3 给 出 了 采用 最 近 最 久未 使 用 更 新 算法 的 页 面 置换 过 程 。 


表 5-3 ”最近 最 久未 使 用 更 新 算法 页 面 置换 表 


页 面 走向 4 3 2 1 4 Ey 上 4 3 2 1 5 
物理 块 0# 4 4 4 4 4 4 4 4 4 4 4 § 
物理 块 1# 3 3 3 3 3 3 e 六 3 3 3 
物理 块 2# 2 2 要 2 5 EG a 人 1 1 
物理 块 3# 1 1 1 | | 1 2 2 2 
是 否 缺 页 六 Y 区 Y 家 1 学 区 


从 表 5-3 中 可 以 看 出 ,此 算法 产生 8 次 缺 页 ,其 中 前 4 次 的 缺 页 情况 和 OPT 算法 一 样 ， 
然而 当 第 一 次 访问 5# 页 时 ,此 算法 看 当前 内 存 中 的 4 个 页 : 4、3、2、1, 谁 是 最 久未 被 访问 过 
的 , 即 从 当前 时 刻 沿 时 间 轴 向 后 看 ,3# 页 是 刚刚 被 访问 过 的 ,而 2# 页 是 最 久未 被 访问 过 
的 ,所 以 2# 页 被 淘汰 ,而 不 管 将 来 是 否 要 访问 2# 页 。 采 用 LRU 算法 时 ,系统 总 共 发 生 了 
8 次 页 面 置换 , 缺 页 率 为 8/12。 

LRU 算法 与 每 个 页 面 最 后 被 访问 的 时 间 有 关 。 该 算法 赋予 每 个 页 个 访问 字段 ,用 
来 记录 一 个 页 面 自 上 次 被 访问 以 来 所 经 历 的 时 间 1, 当 必须 淘汰 一 个 页 面 时 ,此 算法 将 内 存 
中 1 值 最 大 的 页 面 了 予以 淘汰 。 


页 式 和 段 式 内 存 管理 


媳 串 避 


操作 系统 原理 


5.3 段 式 内 存 管理 


分 区 式 管理 和 页 式 管理 时 的 进程 地 址 空间 结构 都 是 线性 的 ,这 就 要 求 对 源 程序 进行 编 
译 、 链 接 时 ,把 源 程序 中 的 主 程序 、 子 程序 、 数 据 区 等 按 线性 空间 的 一 维 地 址 顺序 排列 起 来 。 
这 使 得 不 同 作业 或 进程 之 间 共 享 共 用 子 程序 和 数据 变 得 非常 困难 。 如 果 系 统 不 能 把 用 户 给 
定 的 程序 名 和 数据 块 名 与 这 些 被 共享 程序 和 数据 在 某 个 进程 中 的 虚拟 页 面 对 应 起 来 , 则 不 
可 能 共享 这 些 存 放 在 内 存 页 面 中 的 程序 和 数据 。 为 了 满足 用 户 的 需要 ,更 好 地 实现 共享 和 
保护 ,在 现代 操作 系统 中 引入 了 段 式 内 存 管理 技术 。 


5.3.1 基本 原理 


段 式 内 存 管 理 是 基于 为 用 户 提 供 一 个 方便 灵活 的 程序 设计 环境 而 提出 来 的 。 段 式 内 存 
管理 的 基本 思想 是 : 把 程序 按 内 容 或 过 程 ( 函数) 关系 分 成 段 , 每 个 段 都 有 自己 的 名 称 。 一 
个 用 户 作 业 或 进程 所 包含 的 段 对 应 于 一 个 二 维 线性 虚拟 空间 ,也 就 是 一 个 二 维 虚拟 存储 器 。 
段 式 管理 程序 以 段 为 单位 分 配 内 存 , 然 后 通过 地 址 映射 机 制 把 段 式 虚拟 地 址 转换 成 实际 的 
内 存 物 理 地 址 。 


5.3.2 分 段 内 存 管 理 


1. 段 式 地 址 结构 

分 段 内 存 管 理 系统 把 一 个 进程 的 虚拟 地 址 空间 设计 成 二 维 结构 , 即 段 号 s 和 段 内 相对 
地 址 w。 在 分 页 内 存 管 理 系 统 中 ,被 划分 的 页 号 按 顺序 编号 递增 排列 ,属于 一 维 空间 ,而 分 
段 内 存 管理 系统 中 的 段 号 与 段 号 之 间 是 没有 顺序 关系 的 。 

另外 , 段 的 划分 也 不 像 页 的 划分 那样 具有 相同 的 页 长 , 段 的 长 度 是 不 固定 的 。 每 个 段 定 
义 一 组 逻辑 上 完整 的 程序 或 数据 。 例 如 ,一 个 进程 中 的 程序 和 数据 可 被 划分 为 主 程序 段子 
程序 段 、 数 据 段 与 工作 区 段 。 每 个 段 是 一 个 首 地 址 为 零 的 ,连续 的 一 维 线性 空间 。 根 据 需 
要 , 段 长 可 动态 增长 。 对 段 式 虚拟 地 址 空间 的 访问 包括 两 部 分 : 段 名 和 段 内 地 址 。 

2. 内 存 的 分 配 

分 段 内 存 管理 系统 中 以 段 为 单位 分 配 内 存 , 每 段 分 配 一 个 连续 的 内 存 区 。 由 于 各 段 长 
度 不 等 ,所 以 这 些 存 储 区 大 小 不 等 。 此 外 ,同一 进程 包含 的 各 段 之 间 不 要 求 连续 。 分 段 内 存 
管理 的 内 存 分 配 与 释放 在 作业 或 进程 的 执行 过 程 中 动态 进行 。 首 先 ,分 段 内 存 管 理 程序 为 
一 个 进入 内 存 准 备 执行 的 进程 或 作业 分 配 部 分 内 存 , 以 作为 该 进程 的 工作 区 和 放置 即将 执 
行 的 程序 段 。 随 着 进程 的 执行 ,进程 根据 需要 随时 申请 调 入 新 段 和 释放 旧 段 。 进 程 对 内 存 
区 的 申请 和 释放 可 分 为 以 下 两 种 情况 。 

(1) 当 进程 要 求 调 入 某 一 段 时 ,内 存 中 有 足够 的 空闲 区 满足 该 段 的 内 存 要 求 。 

系统 要 用 相应 的 表格 或 数据 结构 来 管理 内 存 空闲 区 ,以 便 对 用 户 进程 或 作业 的 有 关 程 
序 段 进行 内 存 分 配 和 回收 。 事 实 上 .可 以 采用 和 动态 分 区 式 管理 相同 的 空闲 区 管理 方法 , 即 
把 内 存 中 各 个 空闲 区 按 物 理 地 址 从 低 到 高 排列 或 按 空闲 区 从 小 到 大 或 从 大 到 小 排列 。 与 这 
几 种 空闲 区 自由 链 相 对 应 ,最 先 适 应 算法 、 最 佳 适应 算法 .最 坏 适 应 算法 都 可 用 来 进行 空闲 
区 分 配 。 分 区 管理 时 用 到 的 内 存 回收 方法 也 可 以 在 分 段 内 存 管理 中 使 用 。 


(2) 内 存 中 没有 足够 的 空闲 区 满足 该 段 的 内 存 要 求 。 

此 时 ,分 段 内 存 管理 程序 根据 给 定 的 置换 算法 淘汰 内 存 中 在 今后 一 段 时 间 内 不 再 被 
CPU 访问 的 段 , 即 淘汰 那些 访问 概率 最 低 的 段 。 

3. 地 址 映射 

在 分 段 内 存 管理 系统 中 ,用 户 虽 然 能 够 用 二 维 地 址 来 引用 程序 中 的 对 象 ,但 实际 上 , 物 
理 内 存 仍 然 是 一 维 序列 的 字 节 ,因此 ,必须 定义 一 个 实现 方式 ,以 便 将 二 维 的 用 户 定义 地 址 
映射 为 一 维 物 理 地 址 。 这 个 映射 是 通过 段 表 来 实现 的 。 段 表 的 每 个 条 目 都 有 段 基地 址 和 段 
界限 。 段 基地 址 包含 该 段 在 内 存 中 的 起 始 物理 地 址 ,而 段 界 限 指定 该 段 的 长 度 。 一 个 进程 
的 所 有 段 都 应 该 在 该 进程 的 段 表 中 登记 ,每 个 段 的 信息 是 段 表 中 的 一 条 记录 。 通 常 , 段 表 存 
放 在 内 存 中 ,属于 进程 的 现场 信息 。 为 了 方便 地 找到 运行 进程 的 段 表 ,系统 还 需要 建立 一 个 
段 表 寄存 器 。 段 表 寄 存 器 由 两 部 分 组 成 : 一 部 分 指出 该 段 表 在 内 存 中 的 起 始 地 址 ; 另 一 部 
分 指出 该 段 表 的 长 度 。 

段 地 址 的 转换 过 程 如 图 5-8 所 示 。 
段 表 长 段 表 地 址 

区 B | 段 表 地 址 寄存 器 

段 表 内 存 


( 段 长 ) 


民有 地 (内 存 地址 ) 


base 


逻辑 地 址 - i 


ms 


地 址 错 ， 发 中 断 
图 5-8 分 段 地 址 转换 


一 个 逻辑 地 址 由 两 部 分 组 成 : 段 号 y 和 段 内 地 址 4d。 系统 根据 段 表 地 址 寄存 器 中 的 起 
始 地 址 找到 该 进程 的 段 表 ,以 段 号 为 索引 查找 相应 的 段 表 项 ,得 出 该 段 的 长 度 limit 及 该 段 
在 内 存 中 的 起 始 地 址 base。 然 后 ,将 段 内 地 址 d 与 段 长 limit 进行 比较 。 如 果 d 宇 limit, 则 
表示 地 址 越界 ,系统 产生 地 址 越界 中 断 ,终止 程序 的 执行 ; 如 果 d 过 limit, 则 表示 地 址 合法 ， 
将 段 内 地 址 4 与 该 段 的 内 存 起 始 地 址 base 相 加 ,得 到 所 要 访问 单元 的 内 存 物 理 地 址 。 

4. 分 段 系统 的 优 缺 点 

分 段 系 统 为 用 户 提供 了 一 个 二 维 的 虚拟 地 址 空间 ,反映 了 程序 的 逻辑 结构 ,有 利于 段 的 
动态 增长 及 共享 和 内 存 保护 等 ,这 大 大 地 方便 了 用 户 ,但 仍然 存在 碎片 问题 。 


5.3.3 段 的 共享 与 保护 


段 式 内 存 管理 可 以 方便 地 实现 内 存 信息 共享 和 进行 有 效 的 内 存 保护 。 这 是 因为 段 是 按 
照 逻辑 意义 来 划分 的 ,可 以 按 段 名 来 访问 。 


t 


页 式 和 段 式 内 存 管理 


莉 吕 避 


楷 作 系统 原理 


在 多 道 环境 下 ,常常 有 许多 子 程序 和 应 用 程序 是 被 多 个 用 户 使 用 的 。 特 别 是 在 多 窗口 
系统 中 支持 工具 等 广泛 流行 的 今天 ,被 共享 的 程序 和 数据 的 个 数 和 体积 都 在 急剧 增加 ,有 
寺 会 超过 用 户 程 序 长 度 的 许多 售 。 在 这 种 情况 下 ,如 果 每 个 用 户 进 程 或 作业 都 在 内 存 中 保 
如 它们 共享 程序 和 数据 的 副本 ,就 会 极 大 地 浪费 内 存 空间 。 最 好 的 办 法 是 内 存 中 只 保留 一 
个 副本 供 多 个 用 户 使 用 , 称 为 共享 。 
另外 ,由 于 多 道 环境 中 进程 的 并 发 执行 ,一 段 程序 为 多 个 进程 共享 时 ,有 可 能 出 现 多 次 
同时 重复 执行 该 段 程序 的 情况 。 这 就 要 求 程序 在 执行 过 程 中 , 它 的 指令 和 数据 不 能 被 修改 。 
与 一 个 进程 中 的 其 他 程序 段 一 样 , 共 享 段 有 时 也 要 被 换 出 内 存 。 此 时 ,应 在 段 表 中 设立 相应 
为 共享 位 来 判断 该 段 是 否 正 在 被 某 个 进程 调用 。 一 个 正在 被 某 个 进程 使 用 或 即将 被 某 个 进 
程 使 用 的 共享 段 是 不 应 该 调 出 内 存 的 。 


5.3.4 分 页 与 分 段 管理 的 主要 区 别 


通过 前 面 的 介绍 ,可 以 发 现 分 页 与 分 段 有 许多 相似 之 处 ,如 二 者 在 内 存 中 都 不 是 整体 连 
续 的 ,但 是 ,二 者 在 概念 上 是 完全 不 同 的 ,具体 表现 在 以 下 三 个 方面 。 

(1) 页 是 信息 的 物理 单位 , 段 是 信息 的 逻辑 单位 。 

分 页 是 为 了 实现 离散 分 配方 式 , 以 减少 内 存 碎片 ,提高 内 存 利用 率 。 或 者 说 ,分 页 仅仅 
是 由 于 系统 管理 的 需要 ,而 不 是 用 户 的 需要 。 段 则 是 信息 的 逻辑 单位 , 它 含 有 一 组 意义 相对 
完整 的 信息 。 分 段 的 目的 是 更 好 地 满足 用 户 的 需要 。 

(2) 页 的 大 小 是 由 系统 确定 的 ,而 段 的 长 度 是 由 用 户 程序 确定 的 。 

系统 把 逻辑 地 址 划分 成 页 号 和 页 内 地 址 两 部 分 ,每 个 系统 只 能 有 一 种 大 小 的 页 面 ; 而 
段 的 长 度 是 不 固定 的 ,其 长 度 取决 于 用 户 的 程序 。 通 常 由 编译 程序 在 对 源 代码 进行 编译 时 ， 
根据 信息 的 性 质 来 划分 段 。 

(3) 分 页 的 进程 地 址 空间 是 一 维 的 , 即 单一 的 线性 空间 ; 而 分 段 的 进程 地 址 空间 是 二 
维 的 ,由 段 号 和 段 内 地 址 两 部 分 组 成 。 


5.3.5 段 页 式 内 存 管 理 


段 式 内 存 管理 为 用 户 提 供 了 一 个 三 维 的 虚拟 地 址 空间 ,反映 了 程序 的 逻辑 结构 ,有 利于 
段 的 动态 增长 及 共享 和 内 存 保护 等 ,这 大 大 地 方便 了 用 户 ,但 是 存在 碎片 问题 。 页 式 内 存 管 
理 系 统 有 效 地 克服 了 碎片 ,提高 了 存储 器 的 利用 率 。 内 存 管理 的 目的 主要 是 方便 用 户 的 程 
序 设计 和 提高 内 存 的 利用 率 。 因 此 ,把 段 式 内 存 管 理 与 页 式 内 存 管 理 两 种 方式 结合 起 来 使 
其 互相 取长补短 是 一 种 更 好 的 内 存 管 理 方式 , 段 页 式 内 存 管 理 方式 被 提出 。 

1. 基本 原理 

段 页 式 内 存 管理 时 ,一 个 进程 仍然 拥有 一 个 自己 的 二 维 地 址 空间 ,这 与 段 式 内 存 管 理 方 
式 相同 。 段 页 式 内 存 管 理 方式 将 地 址 空间 划分 为 三 个 部 分 : 段 号 *, 页 号 p 和 页 内 地 址 4d， 
如 图 5-9 所 示 。 


首先 ,一 个 进程 中 所 包含 的 具有 独立 逻辑 功能 的 程 
[二 序 或 数据 仍然 被 划分 为 段 ,并 有 各 自 的 段 号 s。 其 次 ,对 
于 段 ;中 的 程序 或 数据 ,按照 一 定 的 大 小 将 其 划分 为 不 

图 5-9 段 页 式 内 存 管理 地 址 结构 ” 同 的 页 ,最 后 不 足 一 页 的 部 分 仍然 占 一 页 ,这 反映 了 段 


页 式 内 存 管理 中 的 页 式 特点 。 


在 由 三 个 部 分 组 成 的 逻辑 地 址 中 ,程序 员 可 见 的 仍然 是 段 号 y* 和 段 内 相对 地 址 w。p 


和 d 是 由 地 址 映射 机 构 把 ww 的 高 几 位 解释 成 页 号 ,以 及 把 剩余 的 低位 解释 为 页 内 
而 得 到 的 。 
由 于 虚拟 空间 的 最 小 单位 是 页 而 不 是 段 ,所 以 内 存 可 用 区 被 划分 为 若干 个 大 小 
页 面 , 且 每 段 所 拥有 的 程序 和 数据 在 内 存 中 可 以 分 开 存放 。 分 段 的 大 小 也 不 再 受 内 
区 的 限制 。 
2. 段 表 和 页 表 


地 址 a 


相等 的 
存 可 用 


为 了 实现 段 页 式 内 存 管理 ,系统 将 为 每 个 作业 或 进程 建立 一 个 段 表 ,管理 内 存 的 分 配 与 


释放 、 缺 段 处 理 、 存 储 保护 和 地 址 映射 等 。 另 外 ,由 于 一 个 段 又 被 划分 为 若干 页 ,每 个 
须 建 立 一 张 页 表 , 把 段 中 的 虚 页 变换 为 内 存 中 的 实际 页 面 。 与 页 式 内 存 管理 相同 ,页 
要 有 实现 缺 页 中 断 处 理 和 页 面 保护 等 功能 的 表 项 。 由 于 在 段 页 式 内 存 管理 中 ,页 表 
于 进程 ,而 是 属于 某 个 段 , 所 以 , 段 表 中 应 单独 指出 该 段 所 对 应 的 页 表 的 起 始 地 址 和 
度 。 段 页 式 内 存 管理 中 的 段 表 、 页 表 和 内 存 的 关系 如 图 5-10 所 示 。 


第 0 段 页 面 


段 又 必 
表 中 也 
不 再 属 
页 表 长 


段 表 地 址 寄存 器 「 贡 号 [其 他 [页 而 ] 
段 表 长 度 | 起 始 地 址 
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图 5-10 段 页 式 内 存 管理 中 的 段 表 、 页 表 与 内 存 的 关系 


3. 动态 地 址 转换 


进程 的 段 表 和 页 表 。 因 此 ,在 段 页 式 内 存 管理 系统 中 ,要 对 内 存 中 的 指令 或 数据 进行 
取 , 至 少 需要 访问 内 存 三 次 以 上 。 


在 使 用 段 页 式 内 存 管 理 的 计算 机 系统 中 ,一 般 都 在 内 存 中 开辟 出 一 块 固定 的 区 域 存放 


一 次 在 


第 一 次 访问 内 存 : 由 段 表 地 址 寄存 器 得 到 段 表 的 起 始 地 址 ,从 而 访问 段 表 ,由 此 取出 对 


应 段 的 页 表 起 始 地 址 。 
第 二 次 访问 内 存 : 根据 页 表 的 起 始 地 址 访问 页 表 , 得 到 所 要 访问 的 物理 地 址 。 
第 三 次 访问 内 存 : 根据 得 到 的 物理 地 址 ,访问 内 存 中 真正 的 物理 单元 。 


由 此 可 知 ,只 有 在 访问 了 段 表 和 页 表 之 后 ,才能 真正 访问 内 存 中 实际 的 物理 单元 ,这 将 


页 式 和 段 式 内 存 管理 


其 器 避 


握 作 系统 原理 


使 CPU 执行 指令 的 速度 大 大 降低 。 
为 了 提高 地 址 转换 速度 ,设置 快速 联想 寄存 器 就 显得 比 段 式 内 存 管理 或 页 式 内 存 管 理 
时 更 加 需要 。 在 快速 联想 寄存 器 中 ,存放 当前 最 常用 的 段 号 ;、 页 号 p 和 对 应 的 内 存 页 面 与 
其 他 控制 信息 。 当 要 访问 内 存 空间 某 一 单元 时 ,可 在 通过 段 表 、 页 表 进 行内 存 地 址 查找 的 同 
时 ,根据 快速 联想 寄存 器 查找 其 段 号 和 页 号 。 如 果 要 访问 的 段 或 页 在 快速 联想 寄存 器 中 , 则 
系统 不 再 访问 内 存 中 的 段 表 、 页 表 . 而 是 直接 把 快速 联想 寄存 器 中 的 值 与 页 内 相对 地 址 4 
拼接 起 来 ,得 到 相应 的 物理 地 址 。 段 页 式 内 存 管 理 的 地 址 转换 机 构 如 图 5-11 所 示 。 
段 表 地 址 寄存 器 虚拟 地 址 联想 寄存 器 
段 表 长 度 | 起 始 地 址 | 匡 汉 医治 区: 2 


slplp 


段 表 


/ +d 
|] 


页 面 


图 5-11 段 页 式 内 存 管理 系统 地 址 转换 


总 之 ,因为 段 页 式 内 存 管 理 是 段 式 内 存 管 理 和 页 式 内 存 管理 方案 结合 而 成 的 ,所 以 同时 
具有 它们 的 优点 。 


5.4 虚拟 内 存 


虚拟 内 存 是 内 存 管理 技术 的 一 个 极其 实用 的 创新 。 它 是 一 段 程序 持续 监控 着 所 有 物理 
内 存 中 的 代码 段 和 数据 段 ,并 保证 它们 在 运行 中 的 效率 以 及 可 靠 性 ,对 于 每 个 用 户 层 的 进程 
分 配 一 段 虚拟 内 存 空间 。 当 进程 建立 时 ,不 需要 在 物理 内 存 之 间 搬 移 数据 ,数据 存储 于 磁盘 
内 的 虚拟 内 存 空 间 , 也 不 需要 为 该 进程 配置 主 内 存 空间 ,只 有 当 该 进程 被 调用 的 时 候 才 会 被 
加 载 到 主 内 存 。 

可 以 想象 一 个 很 大 的 程序 , 当 它 执行 时 被 操作 系统 调用 ,其 运行 需要 的 内 存 数据 都 被 存 
放 到 磁盘 内 的 虚拟 内 存 , 只 有 需要 用 到 的 部 分 才 被 加 载 到 主 内 存 内 部 运行 。 


5.4.1 虚拟 内 存 


1. 虚拟 内 存 的 引入 

1) 常规 存储 器 管理 方式 的 特征 

(1) 一 次 性 : 作业 全 部 装 入 内 存 后 才能 开始 运行 。 

(2) 驻 留 性 : 作业 装 入 内 存 后 , 便 一 直 驻 留 在 内 存 中 ,直至 作业 运行 结束 。 

2) 局 部 性 原理 

程序 在 执行 时 呈现 出 局 部 性 规律 , 即 在 一 段 时 间 内 ,整个 程序 的 执行 仅 限于 程序 中 的 某 
一 部 分 。 相 应 地 ,执行 所 访问 的 存储 空间 也 局 限于 某 个 内 存 区 域 。 局 部 性 原理 又 表现 为 : 
时 间 局 部 性 和 空间 局 部 性 。 


(1) 时 间 局 部 性 

现象 : 如 果 程 序 中 某 条 指令 一 旦 执行 , 则 不 久 后 该 指令 可 能 再 次 执行 ;如果 某 数 据 被 
访问 过 , 则 不 久 后 该 数据 可 能 再 次 被 访问 。 

原因 : 程序 中 存在 大 量 的 循环 操作 。 

(2) 空间 局 部 性 

现象 : 一 旦 程序 访问 了 某 个 存储 单元 ,在 不 久之 后 ,其 附近 的 存储 单元 将 被 访问 , 即 程 
序 在 一 段 时 间 内 访问 的 地 址 ,可 能 集中 在 一 定 的 范围 之 内 。 

原因 : 程序 的 顺序 执行 。 

在 计算 机 的 应 用 中 ,用 户 作 业 的 地 址 空间 或 多 个 作业 的 地 址 空间 总 和 超过 主 存 可 用 空 
间 时 ,是 不 是 就 不 可 以 在 计算 机 上 运行 呢 ? 是 不 是 一 定 要 全 部 装 入 内 存 才能 运行 呢 ? 

根据 程序 运行 的 局 部 性 原理 ,答案 为 : 不 是 。 

鉴于 局 部 性 原理 的 原因 ,将 作业 的 全 部 内 容 始终 装 入 主 存 是 对 资源 的 极 大 浪费 。 因 此 ,可 
以 将 作业 中 最 近 经 常 需要 访问 的 某 一 部 分 代码 和 数据 装 入 主 存 , 其 余部 分 暂 不 装 入 。 这 样 , 作 
业 的 全 部 内 容 存放 在 辅 存 上 , 主 存 只 存放 一 部 分 。 用 户 的 作业 大 小 就 不 受 主 存 大 小 的 限制 了 。 

从 效果 上 看 ,计算 机 系统 提供 了 一 个 存储 容量 比 实际 内 存 大 得 多 的 存储 器 。 用 户 也 认 
为 自己 在 一 个 比 主 存 容量 大 的 系统 中 运行 作业 ,这 种 存储 系统 就 是 虚拟 存储 器 。 

2. 虚拟 存储 器 的 定义 

虚拟 存储 器 是 指 具 有 请 求 调 入 功能 和 置换 功能 ,能 从 逻辑 上 对 内 存 容 量 加 以 扩充 的 一 
种 存储 器 系统 。 

虚拟 存储 器 的 逻辑 容量 是 内 存 容量 和 外 存 容 量 之 和 ,最 大 容量 由 计算 机 的 地 址 结构 决 
定 。 虚 拟 存储 器 的 运行 速度 接近 内 存 , 成 本 接近 外 存 。 

3. 虚拟 存储 器 的 工作 原理 

(1) 在 分 页 和 分 段 系统 的 基础 上 ,增加 了 请 求 调 页 、 请 求 调 段 和 页 面 置换 、 段 置换 功能 
所 形成 的 页 式 和 段 式 虚拟 存储 系统 。 

(2) 它 允 许 只 装 入 部 分 页 面 和 段 的 程序 或 数据 , 便 启动 运行 作业 。 如 果 以 后 需要 不 在 
内 存 中 的 程序 或 数据 时 ,再 通过 请 求 调 页 、 请 求 调 段 功能 和 页 面 置换 、 段 置换 功能 ,陆续 把 即 
将 运行 的 页 面 、 段 调 入 内 存 , 同 时 把 暂 不 运行 的 页 面 和 段 置 换 到 外 存 上 。 

(3) 置换 时 以 页 面 和 段 为 单位 进行 。 

4. 虚拟 存储 器 的 特征 

(1) 多 次 性 : 作业 分 多 次 调 入 内 存 , 是 虚拟 存储 器 特有 的 功能 。 

(2) 对 换 性 : 允许 作业 在 执行 的 过 程 中 换 入 、 换 出 ,从 而 提高 内 存 的 利用 率 。 

(3) 虚拟 性 : 从 逻辑 上 扩大 内 存 容量 ,使 用 户 看 到 的 内 存 容量 远大 于 实际 的 内 存 容 


5.4.2 请 求 分 页 式 内 存 管理 


在 页 式 内 存 管理 系统 中 ,允许 一 个 作业 程序 只 装 入 部 分 页 面 即 可 投入 运行 ,需要 信息 时 
动态 调 入 ,这 种 装 入 信息 的 策略 称 为 请 调 策 略 ,实现 这 种 策略 的 系统 称 为 请 求 分 页 式 内 存 管 
理 系 统 。 

进程 在 运行 过 程 中 必然 会 遇 到 所 需要 的 代码 或 数据 不 在 主 存 的 情况 ,因此 ,系统 必须 解 
决 以 下 两 个 问题 。 


页 式 和 段 式 内 存 管理 


攻 口 避 


妮 作 系统 原理 


问题 1: 怎样 确定 所 访问 的 页 面 在 不 在 内 存 ? 
问题 2: 如 果 确 认 所 要 访问 的 页 面 不 在 主 存 时 如 何 处 理 ? 


1. 页 表 机 制 
请 求 分 页 式 内 存 管理 方式 中 的 页 表 的 作用 是 将 用 户 地 址 空间 的 逻辑 地 址 转换 为 内 存 空 


间 的 物理 地 址 。 由 于 请 求 分 页 的 特殊 性 ,即将 程序 的 一 部 分 调 入 内 存 , 另 一 部 分 仍 在 外 存 ， 
因此 页 表 的 结构 有 所 不 同 ,如 图 5-12 所 示 。 


页 号 物理 块 号 状态 位 已 访问 字段 A 修改 位 M 外 存 地址 


图 5-12 请 求 分 页 式 内 存 管理 方式 的 页 表 项 字段 


(1) 状态 位 已 : 指示 本 页 是 否 已 经 调 入 内 存 。 

(2) 访问 字段 A: 记录 本 页 在 一 段 时 间 内 被 访问 的 次 数 或 最 近 未 被 访问 的 时 间 。 

(3) 修改 位 M: 表示 本 页 在 调 入 内 存 后 是 否 被 修改 过 。 如 果 被 修改 过 , 则 在 换 出 时 需 
要 重新 写 到 外 存 上 。 

(4) 外 存 地 址 : 指出 本 页 在 外 存 上 的 存储 地 址 。 

2. 缺 页 中 断 机 构 

在 请 求 分 页 式 内 存 管理 系统 中 ,每 当 所 要 访问 的 页 不 在 内 存 时 , 便 产 生 一 次 缺 页 中 断 ， 
请 求 操作 系统 将 所 缺 之 页 调 入 内 存 。 执 行 过 程 如 图 5-13 所 示 。 


一 一 | 启动 要 处 理 的 指令 了 
计算 有 效 地 址 
取 下 一 条 指令 仿 
取出 页 号 国 
访问 内 存 ， 完 成 该 指令 由 


按 外 存 地 址 读 入 
所 需 的 页 面 


调整 页 表 及 
存储 分 块 表 
1 
|_「 重新 局 动 

被 中 断 的 指令 


图 5-13 ”指令 执行 步骤 与 缺 页 中 断 处 理 过 程 


3. 地 址 转换 机 构 
在 分 页 系统 地 址 转换 的 基础 上 ,为 了 实现 虚拟 存储 器 而 增加 了 产生 和 处 理 缺 页 中 断 、 内 


存 页 的 换 出 等 功能 。 转 换 过 程 如 下 。 

1) 检索 快 表 试 图 找到 所 要 访问 的 页 

若 在 快 表 中 找到 和 欲 访 问 的 页 号 ,修改 页 表 项 中 的 访问 位 A。 对 于 写 指令 ,将 修改 位 置 
1。 然 后 利用 页 表 项 中 给 出 的 物理 块 号 和 页 内 地 址 ,形成 物理 地 址 。 地 址 转换 结束 。 若 在 快 
表 中 未 找到 和 欲 访问 的 页 号 , 则 转 至 (2), 到 页 表 中 查找 。 

2) 检索 内 存 中 的 页 表 

从 页 表 的 始 址 寄存 器 中 获取 页 表 所 在 的 物理 块 号 。 在 内 存 中 找到 页 表 , 检 查 页 表 项 中 
的 状态 位 ,确定 其 是 否 已 经 装 入 内 存 。 如 果 已 经 装 入 内 存 , 则 将 该 页 填 入 快 表 中 。 若 快 表 
项 已 满 , 则 选择 调 出 的 页 ,并 修改 页 表 项 。 如 果 没 有 装 入 内 存 , 则 产生 缺 页 中 断 ,请 求 操作 系 
统 将 该 页 调 入 内 存 中 ,并 加 入 快 表 中 ,修改 页 表 项 。 修 改 页 表 项 中 的 访问 位 A。 对 于 写 指 
令 , 将 修改 位 设置 为 1。 然 后 利用 页 表 项 中 给 出 的 物理 块 号 和 页 内 地 址 ,形成 物理 地 址 ,地 
址 转换 结束 。 如 图 5-14 所 示 为 请 求 分 页 式 地 址 转换 过 程 。 


越界 中 断 
页 表 始 址 寄存 器 逻辑 地 址 
页 表 起 始 地 址 页 号 (2) | 页 内 地 址 (452) 
| 
| 页 号 块 号 


| 


回 写 快 表 | 其 S) | 其 内 地 直 (452) 


物理 地 址 


图 5-14 请 求 分 页 式 地 址 转换 过 程 


5.4.3 请 求 分 段 式 内 存 管理 


在 分 段 基础 上 建立 的 请 求 分 段 式 内 存 管理 是 以 分 段 为 单位 进行 换 入 、 换 出 的 。 当 所 访 
问 的 段 不 在 内 存 中 时 ,可 请 求 操作 系统 将 所 缺 的 段 调 入 内 存 中 。 

1. 段 表 机 制 

在 请 求 分 段 式 内 存 管 理 中 所 需要 的 主要 数据 结构 是 请 求 式 段 表 。 在 该 表 中 除了 具有 
请 求 分 页 式 内 存 管理 方式 中 页 表 中 的 访问 字段 A、 修 改 位 M、 状 态 位 已 和 外 存 地 址 4 个 
字段 外 ,还 增加 了 存 取 方式 字段 和 增补 位 。 这 些 字段 供 程序 在 调 进 、 调 出 时 参考 ,如 图 5-15 


段 号 段 长 | 段 起 始 地 址 | 存 取 方 式 | 状态 位 P | 访问 字段 A| 修改 位 M | 增补 位 | 外 存 地 址 


图 5-15 请 求 分 段 式 内存 管 理 方式 的 段 表 项 字段 


页 式 和 禾 式 内 存 管理 


媳 品 避 


握 作 系统 原理 


在 段 表 项 中 ,除了 段 号 、 段 长 、 段 起 始 地 址 外 ,还 增加 了 以 下 字段 。 

(1) 存 取 方式 : 由 于 应 用 程序 中 的 段 是 信息 的 逻辑 单位 ,可 根据 该 信息 的 属性 对 它 实 
施 保护 ,因此 在 段 表 中 增设 了 存 取 方式 字段 。 如 果 该 字段 为 两 位 , 则 存 取 属性 是 只 执行 、 只 
读 和 允许 读 \ 写 。 

(2) 状态 位 P: 指示 本 段 是 否 已 经 调 入 内 存 。 

(3) 访问 字段 A: 记录 本 段 在 一 段 时 间 内 被 访问 的 次 数 或 最 近 未 被 访问 的 时 间 。 

(4) 修改 位 M: 表示 本 段 在 调 入 内 存 后 是 否 被 修改 过 。 如 果 被 修改 过 , 则 在 换 出 时 需 
要 重新 写 到 外 存 上 。 

(5) 增补 位 : 这 是 请 求 分 段 式 内 存 管理 中 所 特有 的 字段 ,用 于 表示 本 段 在 运行 过 程 中 
是 否 做 过 动态 增长 。 

(6) 外 存 地 址 : 指出 本 页 在 外 存 上 的 存储 地 址 。 

2. 缺 段 中 断 机 构 

在 请 求 分 段 式 内 存 管 理 系 统 中 采用 的 是 请 求 调 段 策略 。 每 当 发 现 运行 进程 所 要 访问 的 
段 尚未 调 入 内 存 时 , 便 由 缺 段 机 构 产 生 一 个 缺 段 中 断 信号 ,进入 操作 系统 后 ,由 缺 段 中 断 处 
理 程序 将 所 需要 的 段 调 入 内 存 。 

与 缺 页 中 断 机 构 类 似 , 缺 段 中 断 机 构 同 样 需要 在 一 条 指令 的 执行 期 间 产 生 和 处 理 中 断 ， 
以 及 在 一 条 指令 执行 期 间 , 可 能 产生 多 次 缺 段 中 断 。 但 是 ,由 于 分 段 是 信息 的 逻辑 单位 , 因 
此 不 可 能 出 现 一 条 指令 被 分 割 在 两 个 分 段 中 ,以 及 一 组 信息 被 分 割 在 两 个 分 段 中 的 情况 。 

3. 地 址 转换 机 构 

请 求 分 段 式 内 存 管理 系统 中 的 地 址 转换 机 构 是 在 分 段 系 统 地 址 转换 机 构 的 基础 上 形成 
的 。 因 此 被 访问 的 段 并 不 是 全 都 在 内 存 中 的 。 因 此 ,在 地 址 转换 时 ,如 果 发 现 所 要 访问 的 段 
不 在 内 存 中 时 ,必须 先 将 所 缺 的 段 调 入 内 存 , 并 修改 段 表 ,然后 才能 利用 段 表 进 行 地 址 转换 。 


小 结 


用 户 程 序 必须 装 入 到 内 存 中 才能 运行 。 页 式 内 存 管理 技术 允许 进程 的 物理 地 址 空间 不 
连续 ,这 样 ,可 以 把 一 个 程序 分 散在 各 个 空闲 的 物理 块 中 , 它 既 不 需要 移动 内 存 中 原 有 的 信 
息 , 又 解决 了 外 部 碎片 的 问题 ,从 而 提高 了 内 存 的 利用 率 。 

当 内 存 的 总 需求 超出 实际 内 存 容量 时 ,为 了 释放 内 存 块 给 新 的 页 面 ,需要 页 面 更 新 算 
法 。 最 优 更 新 算法 需要 知道 程序 未 来 的 页 面 走向 ,这 实际 上 不 可 行 ,因此 仅 有 理论 价值 。 
FIFO 是 最 容易 实现 的 ,但 性 能 不 是 很 好 。LRU 是 OPT 的 近似 算法 ,但 实际 使 用 时 要 有 硬 
件 的 支持 和 软件 的 开销 。 

段 式 内 存 管理 方式 从 逻辑 上 划分 程序 ,从 而 满足 了 用 户 的 需要 ,更 好 地 实现 共享 和 保 
护 , 同 时 也 避免 了 碎片 的 出 现 , 更 有 效 地 利用 了 内 存 空间 。 
虚拟 存储 器 技术 使 得 请 求 分 页 式 内 存 管 理 和 请 求 分 段 式 内 存 管 理 方式 得 到 了 广泛 的 


第 6 章 VO 管理 


现代 的 计算 机 系统 都 连接 了 很 多 的 外 部 设备 ,通常 统称 1/O 设备 。1/O 设备 管理 是 操 
作 系 统 中 最 繁杂 ,内 容 最 多 的 部 分 。I/O 管理 包括 很 多 问题 ,其 中 包括 缓冲 管理 ,设备 分 配 、 
虚拟 设备 ,磁盘 管理 等 。 我 们 将 会 看 到 1/O 设备 的 管理 既 要 考虑 I/O 设备 的 效率 问题 ,也 
要 考虑 安全 性 问题 。 


6.0 问题 导入 


计算 机 系统 中 的 1/O 设备 越 来 越 多 ,系统 如 何 管理 这 些 设 备 呢 ? 如 何 使 这 些 设 备 都 得 
到 高 效 的 利用 呢 ? 当 多 个 进程 共享 设备 时 ,如 果 管 理 不 当 就 会 造成 错误 。 例 如 ,如 果 系 统 中 
只 有 一 台 打 印 机 ,多 个 进程 都 需要 打印 ,A 进程 提交 打印 任务 的 同时 B 进程 也 提交 打印 任 
务 , 很 有 可 能 A 进程 和 了 BB 进程 的 结果 交叉 输出 ,这 个 输出 结果 是 混乱 无 序 的 ,那么 该 怎么 解 
决 呢 ? 


6.1 1/O 管理 概述 


计算 机 系统 中 除了 CPU 和 内 存 之 外 的 其 他 大 部 分 设备 称 为 外 部 设备 。 因 为 外 部 设备 
的 使 用 需要 和 主机 系统 进行 数据 的 交互 ,一 般 以 CPU 为 中 心 ,根据 数据 的 流动 方向 可 以 将 
设备 分 为 输入 设备 和 输出 设备 ,统称 1/O 设备 。I/O 设备 种 类 繁多 ,从 操作 系统 的 角度 出 发 
可 以 进行 如 下 的 分 类 : 按照 数据 的 传输 速率 可 以 分 为 低速 设备 .中 速 设备 和 高 速 设备 ; 按 
照 数 据 传输 单位 可 以 分 为 块 设备 和 字符 设备 ; 按照 设备 的 共享 情况 可 以 分 为 独占 设备 、 共 
享 设备 和 虚拟 设备 。 

由 于 不 同 1/O 设备 之 间 性 能 差别 非常 大 ,而 且 控 制 方式 也 不 完全 相同 ,所 以 一 般 I/O 
设备 并 不 是 直接 与 主机 相连 ,而 是 通过 设备 控制 器 、 通 道 与 主机 系统 相连 。 下 面 分 别 介 绍 控 
制 器 .通道 。 

设备 控制 器 是 连接 I/O 设备 和 主机 的 中 间接 口 , 用 来 控制 IO 和 主机 之 间 的 数据 交 
换 。 设 备 控制 器 可 以 接收 主机 发 来 的 命令 ,从 而 控制 外 围 设备 ,避免 了 主机 直接 处 理 繁杂 的 
外 围 设备 事务 。 设 备 控制 器 是 可 编 址 的 设备 ,通常 每 个 地 址 对 应 一 个 LO 设备 。 通 过 给 设 
备 控制 器 分 配 多 个 地 址 可 以 实现 设备 控制 器 对 IO 设备 的 一 对 多 控制 。 

由 于 1/O 设备 的 差别 很 大 ,所 以 设备 控制 器 也 不 完全 相同 。 大 体 可 以 分 为 两 类 : 简单 
的 设备 控制 器 和 主要 用 于 控制 块 设备 的 复杂 控制 器 ,通常 控制 器 以 电路 模块 的 形式 放置 在 
微型 计算 机 和 PC 的 主板 上 ,复杂 控制 器 以 印刷 电路 卡 的 形式 单独 存在 。 


所 作 系 统 原 理 


设备 控制 器 一 般 包括 控制 器 与 主机 的 接口 控制 器 和 设备 的 接口 和 控制 器 本 身 的 1/O 
部 分 。 控 制 器 和 主机 的 数据 交换 主要 是 通过 数据 线 . 地 址 线 和 控制 线 来 实现 。 控 制 器 中 包 
含 控制 寄存 器 ,状态 寄存 器 和 数据 寄存 器 三 类 寄存 器 ,用 来 存储 需要 完成 的 通信 类 型 .IO 
设备 的 状态 和 通信 传输 的 数据 。 设 备 控制 器 上 包含 一 个 或 多 个 设备 接口 ,每 个 接口 控制 一 
个 设备 ,每 个 接口 都 包含 数据 ,状态 和 控制 三 种 类 型 的 信号 ,设备 控制 器 根据 I/O 逻辑 系统 
确定 主机 发 来 的 地 址 信号 所 对 应 的 设备 来 完成 通信 。 

设备 控制 器 中 的 W/O 逻辑 用 于 实现 对 设备 的 控制 。 主 机 利用 该 逻辑 向 控制 器 发 送 IO 
命令 ; IO 逻辑 对 收 到 的 命令 进行 译 码 。 每 当 CPU 要 启动 一 个 设备 时 ,一 方面 将 启动 命令 
发 送 给 控制 器 ; 另 一 方面 又 同时 通过 地 址 线 把 地 址 发 送 给 控制 器 ,由 控制 器 的 I/O 逻辑 对 
收 到 的 地 址 进行 译 码 ,再 根据 所 译 出 的 命令 对 所 选 设备 进行 控制 。 设 备 控制 器 的 结构 如 
图 6-1 所 示 。 
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图 6-1 设备 控制 器 的 结构 


如 上 所 述 , 设 备 控制 器 处 在 外 部 设备 和 主机 之 间 完 成 控制 功能 。 控 制 器 需要 完成 的 功 
能 主要 包括 : 通过 1/O 逻辑 部 分 接收 和 识别 主机 发 送 的 命令 ,除了 需要 使 用 命令 寄存 器 来 
存储 主机 发 送 的 命令 之 外 ,还 需要 有 相应 的 命令 译 码 器 实现 命令 译 码 ; 通过 数据 寄存 器 实 
现 主机 和 1/O 设备 之 间 的 数据 交换 ; 使 用 状态 寄存 器 的 内 容 来 标识 和 报告 设备 的 状态 , 主 
机 和 I/O 设备 在 通信 过 程 中 需要 根据 状态 寄存 器 的 内 容 来 确定 可 以 执行 的 操作 ; 通过 地 址 
寄存 器 来 确定 1/O 设备 ,每 个 控制 器 可 以 控制 多 个 IO 设备 ,通过 主机 发 送 给 地 址 寄存 器 
的 地 址 ,设备 控制 器 可 以 确定 主机 指定 的 1/O 设备 ; 通过 缓冲 器 来 提高 整体 数据 传输 的 速 
度 ,1/O 设备 的 速度 一 般 要 比 处 理 机 和 内 存 慢 ,通过 使 用 缓冲 器 可 以 提高 整体 数据 传输 的 速 
度 ; 实现 传输 过 程 中 的 差错 控制 ,通过 对 差错 码 的 设置 ,控制 器 可 以 向 主机 报告 设备 出 错 的 
状况 ,从 而 确保 正确 的 数据 传输 。 

跟 设备 控制 器 的 连接 需要 1/O 设备 提供 相应 的 接口 。 主 要 包括 数据 线 、 地 址 线 和 控制 
线 三 组 信号 线 , 另 外 需要 相应 的 控制 逻辑 以 及 设备 中 的 缓冲 区 和 信号 转换 器 。 

使 用 设备 控制 器 减少 了 处 理 机 对 1/O 的 干预 ,提高 了 系统 的 效率 。 随 着 外 部 设备 的 增 
加 ,处 理 机 的 负担 仍然 很 重 。 为 此 人 们 又 在 主机 和 设备 控制 器 之 间 增 设 了 通道 来 进一步 减 
轻 处 理 机 的 负担 。 通 道 将 I/O 的 组 织 、 管 理 和 结束 处 理 以 通道 程序 的 形式 存储 ,这 样 在 通 
信 的 过 程 中 ,处 理 机 只 需要 发 出 一 个 指令 就 可 以 完成 整个 IO 过 程 ,1/O 结束 后 通道 以 中 断 
的 形式 通知 处 理 机 ,使 得 处 理 机 可 以 从 繁杂 的 I/O 处 理 中 解脱 出 来 。 

实际 上 ,1/O 通道 是 一 种 特殊 的 处 理 机 。 它 具有 执行 IO 指令 的 能 力 ,并 通过 执行 通道 


(1/O 〇 ) 程 序 来 控制 W/O 操作 。 但 1/O 通道 又 与 一 般 的 处 理 机 不 同 ,主要 表现 在 以 下 两 个 方 
面 : 一 是 其 指令 类 型 单一 ,这 是 由 于 通道 硬件 比较 简单 ,其 所 能 执行 的 命令 主要 局 限于 与 
I/O 操作 有 关 的 指令 ; 再 就 是 通道 没有 自己 的 内 存 , 通 道 所 执行 的 通道 程序 是 放 在 主机 的 
内 存 中 的 ,换言之 ,是 通道 与 CPU 共享 内 存 。 

按 通 道 的 工作 方式 ,通道 分 为 选择 通道 、 字 节 多 路 通道 和 数组 多 路 通道 三 种 类 型 。 

(1) 选择 通道 : 这 种 通道 可 以 连接 多 台 快 速 IO 设备 ,但 每 次 只 能 从 中 选择 一 台 设 备 
执行 通道 程序 ,进行 主 存 与 该 设备 之 间 的 数据 传送 。 当 数据 传送 完 后 ,才能 选择 另 一 台 设 
备 。 在 这 种 工作 方式 中 ,数据 传送 以 成 组 方式 进行 ,传送 速率 很 高 ,多 用 于 连接 快速 IO 设 
备 。 但 因 连 接 在 选择 通道 上 的 多 台 设 备 ,只 能 依次 使 用 通道 与 主 存 传送 数据 ,所 以 设备 之 间 
不 能 并 行 工 作 , 且 整个 通道 的 利用 率 不 高 。 

(2) 字 节 多 路 通道 : 这 种 通道 可 以 连接 多 台 慢 速 IO 设备 ,以 交叉 方式 传送 数据 , 即 各 
设备 轮流 使 用 通道 与 主 存 进行 数据 传送 , 且 每 次 只 传送 一 个 字 节 。 因 为 每 次 数据 传送 仅 占 
了 不 同 的 设备 各 自分 得 的 很 短 的 时 间 片 ,所 以 大 大 提高 了 通道 的 利用 率 。 

(3) 数组 多 路 通道 : 数组 多 路 通道 综合 了 选择 通道 和 字 节 多 路 通道 的 优点 , 它 有 多 个 
子 通道 。 既 可 以 像 字 节 多 路 通道 那样 ,执行 多 路 通道 程序 ,使 所 有 子 通道 分 时 共享 总 通道 
又 可 以 像 选择 通道 那样 进行 成 组 数据 的 传送 。 

子 通道 是 指 实现 每 个 通道 程序 所 对 应 的 硬 设备 。 选 择 通道 在 物理 上 可 以 连接 多 台 设 
备 ,但 在 一 段 时 间 内 只 能 执行 一 台 设备 的 通道 程序 , 即 在 逻辑 上 只 能 连接 一 台 设 备 ,所 以 它 
只 包含 一 个 子 通道 。 字 节 多 路 通道 和 数组 多 路 通道 在 物理 上 可 以 连接 多 台 设 备 , 而 且 在 一 
段 时 间 内 可 轮流 执行 多 台 设 备 的 通道 程序 , 即 在 逻辑 上 也 可 以 连接 多 台 设 备 ,所 以 它们 包含 
若干 子 通道 。 需 要 注意 的 是 ,一 个 子 通道 可 以 连接 多 台 设 备 ,但 子 通 道 数 并 不 等 于 物理 上 可 
连接 的 设备 数 ,而 是 该 通道 中 能 够 同时 工作 的 设备 数 。 


6.2 1/O 系统 


6.2.1 VO 系统 结构 


1/O 系统 结构 是 指 1/O 设备 和 主机 系统 的 具体 连接 方式 。 不 同 的 计算 机 系统 使 用 不 同 
的 连接 方式 。 对 于 微型 计算 机 和 PC 一 般 使 用 总 线 结 构 将 外 部 设备 和 主机 相连 ,工作 站 和 
工作 站 以 上 的 系统 使 用 通道 .控制 器 的 结构 连接 。 

总 线 结构 如 图 6-2 所 示 。 早 期 的 微型 计算 机 系统 多 采用 单 总 线 结构 ,总 线 分 为 地 址 总 
线 、 数 据 总 线 和 控制 总 线 三 种 。CPU 和 内 存 直 接连 接 到 总 线 上 ,1/O 设备 是 通过 设备 控制 
器 连接 到 总 线 上 。 随 着 微型 计算 机 和 PC 的 发 展 ,也 出 现 了 多 总 线 结构 ,但 是 总 体 的 连接 方 
式 没有 发 生变 化 。 

目前 常用 的 总 线 有 : ISA 总 线 、EISA 总 线 .VESA 总 线 和 PCI 总 线 。 

ISA(Industrial Standard Architecture) 总 线 标准 是 IBM 公司 于 1984 年 为 推出 PC/AT 
计算 机 而 建立 的 系统 总 线 标准 ,所 以 也 叫 AT 总 线 。 在 80286 至 80486 时 代 应 用 非常 广泛 ， 
以 至 于 奔腾 计算 机 中 还 保留 有 ISA 总 线 插 槽 。 

EISA 总 线 是 1988 年 由 Compaq 等 9 家 公司 联合 推出 的 总 线 标准 。 它 是 在 两 条 ISA 信 
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图 6-2 总 线 1/0 结构 


号 线 之 间 添 加 一 条 EISA 信号 线 。 在 使 用 中 ,EISA 总 线 完全 兼容 ISA 总 线 信 号 。 

VESA(Video Electronics Standard Association) 总 线 是 1992 年 由 60 家 附件 卡 制造 商 
联合 推出 的 一 种 局 部 总 线 , 简 称 为 VL(VESA Local) 总线。 该 总 线 系统 考虑 到 CPU 与 主 存 
和 Cache 的 直接 相连 ,通常 把 这 部 分 总 线 称 为 CPU 总 线 或 主 总 线 , 其 他 设备 通过 VL 总 线 
与 CPU 总 线 相 连 ,所 以 VL 总 线 被 称 为 局 部 总 线 。 

PCI(Peripheral Component Interconnect) 总线 是 当前 最 流行 的 总 线 之 一 , 它 是 由 Intel 
公司 推出 的 一 种 局 部 总 线 。PCI 总 线 主板 插 槽 的 体积 比 原 ISA 总 线 插 槽 还 小 ,其 功能 比 
VESA ISA 有 极 大 的 改善 ,支持 突 发 读 写 操作 ,最 大 传输 速率 可 达 132MB/s, 可 同时 支持 多 
组 外 围 设 备 。PCI 局 部 总 线 不 能 兼容 现 有 的 ISA、EISA、MCA (Micro Channel 
Architecture) 总 线 , 但 它 不 受制 于 处 理 器 ,是 基于 奔腾 等 新 一 代 微 处 理 器 而 发 展 的 总 线 。 

使 用 通道 的 主机 系统 结构 如 图 6-3 所 示 。 由 于 通道 和 控制 器 的 价格 较 高 ,所 以 使 用 比 
较 少 的 通道 和 控制 器 控制 设备 ,这 样 既 可 以 节省 成 本 又 可 以 提高 系统 的 效率 。 通 道 和 控制 
器 虽然 可 以 连接 多 个 控制 器 和 设备 ,但 是 同一 时 间 只 能 使 用 其 中 的 一 个 。 另 外 , 当 通 道 和 控 
制 器 出 现 故 障 后 会 影响 设备 的 使 用 ,这 就 使 得 通道 和 控制 器 成 为 系统 的 瓶颈 。 解 决 瓶 颈 问 
题 最 有 效 的 办 法 是 增加 设备 到 主机 之 间 的 通路 而 不 是 增加 通道 。 


控制 器 1 
1/O 通 道 1 中 


控制 器 2 K 设备 
主机 SEE 


控制 器 3 设备 


控制 器 4 C= 设备 


图 6-3 ”使 用 通道 的 1/O 结构 


1/O 通 道 2 


6.2.2 VO 控制 方式 


操作 系统 的 1/0 控制 方式 是 指 操作 系统 控制 I/O 设备 执行 IO 操作 的 方式 ,主要 有 程 
序 直接 控制 方式 、 中 断 方式 .DMA 方式 和 通道 控制 方式 。 

在 程序 直接 控制 方式 中 ,主机 从 外 部 设备 每 次 读 一 个 字 的 数据 到 存储 器 ,如 图 6-4(a) 所 
示 。 读 入 过 程 中 处 理 机 持续 检测 IO 控制 中 的 状态 寄存 器 ,直到 数据 寄存 器 中 的 数据 准备 
好 ,开始 进行 读 入 操作 。 由 于 CPU 的 速度 和 I/O 设备 的 速度 存在 非常 大 的 差别 ,CPU 绝 大 
多 数 的 时 间 都 处 在 状态 检测 过 程 中 ,造成 了 忙 等 待 现象 , 极 大 地 浪费 了 CPU 资源 。 由 于 
CPU 一 直 处 在 忙 等 待 的 循环 测试 中 ,1/O 设备 无 法 及 时 更 新 状态 位 ,造成 了 优先 级 反 转 现 


象 。 程 序 直接 控制 方式 简单 且 易 于 实现 ,但 是 由 于 CPU 和 IO 设备 只 能 串 行 工作 , 导致 
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图 6-4 ”1/0 控制 方式 


中 断 方式 的 思想 是 ,允许 1/O 设备 主动 打 断 CPU 正在 执行 的 运算 并 请 求 系统 提供 服 
务 , 这 就 使 得 CPU 可 以 从 忙 等 待 中 解放 出 来 ,使 得 CPU 向 I/O 控制 器 发 送 读 命令 后 可 以 
继续 做 其 他 有 用 的 工作 。 如 图 6-4(b) 所 示 ,我 们 从 IO 控制 器 和 CPU 两 个 角度 分 别 来 看 
中 断 驱 动 方式 的 工作 过 程 。 

从 I/O 控制 器 的 角度 来 看 ,1/O 控制 器 从 CPU 接收 一 个 读 命令 ,然后 从 外 围 设备 读数 
据 。 一 旦 数据 读 入 到 该 I/O 控制 器 的 数据 寄存 器 , 便 通 过 控制 线 给 CPU 发 出 一 个 中 断 信 
号 ,表示 数据 已 准备 好 ,然后 等 待 CPU 请 求 该 数据 。I/O 控制 器 收 到 CPU 发 出 的 取 数 据 请 
求 后 ,将 数据 放 到 数据 总 线 上 , 传 到 CPU 的 寄存 器 中 。 至 此 ,本 次 I/O 操作 完成 ,I/O 控制 
器 又 可 以 开始 1/O 操作 。 

从 CPU 的 角度 来 看 ,CPU 发 出 读 命令 ,然后 保存 当前 运行 程序 的 上 下 文 (现场 ,包括 程 
序 计数 器 及 处 理 机 寄存 器 ) , 转 去 执行 其 他 程序 。 在 每 个 指令 周期 的 末尾 ,CPU 检查 中 断 。 
当 有 来 自 1/O 控制 器 的 中 断 时 ,CPU 保存 当前 正在 运行 程序 的 上 下 文 , 转 去 执行 中 断 处 理 
程序 处 理 该 中 断 。 这 时 ,CPU 从 1/O 控制 器 读 一 个 字 的 数据 传送 到 寄存 器 ,并 存 入 主 存 。 
接着 ,CPU 恢复 发 出 1/O 命令 的 程序 (或 其 他 程序 ) 的 上 下 文 , 然 后 继续 运行 。 

中 断 方式 比 程序 直接 控制 方式 有 效 ,但 由 于 数据 中 的 每 个 字 在 存储 器 与 I/O 控制 器 之 
间 的 传输 都 必须 经 过 CPU ,这 就 导致 了 中 断 驱动 方式 仍然 会 消耗 较 多 的 CPU 时 间 。 

DMA( 直 接 存储 器 存 取 ) 方 式 的 基本 思想 是 在 1/O 设备 和 内 存 之 间 开 辟 直 接 的 数据 交 
换 通 路 ,彻底 “解放 "CPU。DMA 方式 的 特点 是 : 基本 单位 是 数据 块 。 所 传送 的 数据 ,是 从 设 
备 直接 送 入 内 存 的 ,或 者 相反 。 仅 在 传送 一 个 或 多 个 数据 块 的 开始 和 结束 时 , 才 需 CPU 干预 ， 
整 块 数据 的 传送 是 在 DMA 控制 器 的 控制 下 完成 的 。 图 6-5 列 出 了 DMA 控制 器 的 组 成 。 
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图 6-5 DMA 控制 器 的 组 成 


为 了 实现 在 主机 与 控制 器 之 间 成 块 数据 的 直接 交换 ,必须 在 DMA 控制 器 中 设置 如 下 
4 类 寄存 器 : @@ 命 令 / 状 态 寄存 器 (CR), 用 于 接收 从 CPU 发 来 的 1/O 命令 或 有 关 控 制 信息 ， 
或 设备 的 状态 。 回 内 存 地 址 寄存 器 (MAR) ,在 输入 时 , 它 存 放 把 数据 从 设备 传送 到 内 存 的 
起 始 目 标 地 址 ; 在 输出 时 , 它 存放 由 内 存 到 设备 的 内 存 源 地 址 。@ 数 据 寄 存 器 (DR), 用 于 
暂 存 从 设备 到 内 存 , 或 从 内 存 到 设备 的 数据 。@ 数 据 计 数 器 (DC) ,存放 本 次 CPU 要 读 或 写 
的 字 ( 节 ) 数 。 

如 图 6-4(c) 所 示 ,DMA 方式 的 工作 过 程 是 : CPU 读 写 数据 时 , 它 给 1/O 控制 器 发 出 一 
条 命令 ,启动 DMA 控制 器 ,然后 继续 其 他 工作 。 之 后 CPU 就 把 控制 操作 委托 给 DMA 控 
制 器 ,由 该 控制 器 负责 处 理 。DMA 控制 器 直接 与 存储 器 交互 ,传送 整个 数据 块 ,每 次 传送 
一 个 字 ,这 个 过 程 不 需要 CPU 参与 。 当 传送 完成 后 ,DMA 控制 器 发 送 一 个 中 断 信 号 给 处 
理 器 。 因 此 只 有 在 传送 开始 和 结束 时 才 需 要 CPU 的 参与 。DMA 控制 方式 与 中 断 驱动 方 
式 的 主要 区 别 是 中 断 驱 动 方式 在 每 个 数据 需要 传输 时 中 断 CPU ,而 DMA 控制 方式 则 是 在 
所 要 求 传送 的 一 批 数 据 全 部 传送 结束 时 才 中 断 CPU; 此 外 ,中 断 驱 动 方式 数据 传送 是 在 中 
断 处 理 时 由 CPU 控制 完成 的 ,而 DMA 控制 方式 则 是 在 DMA 控制 器 的 控制 下 完成 的 。 

前 面 已 经 介绍 过 ,1/O 通道 是 指 专门 负责 输入 /输出 的 处 理 机 。I/O 通道 方式 是 DMA 
方式 的 发 展 , 它 可 以 进一步 减少 CPU 的 干预 , 即 把 对 一 个 数据 块 的 读 ( 或 写 ) 为 单位 的 干 
预 ,减少 为 对 一 组 数据 块 的 读 (或 写 ) 及 有 关 的 控制 和 管理 为 单位 的 干预 。 同 时 ,又 可 以 实现 
CPU 、 通 道 和 1/O 设备 三 者 的 并 行 操作 ,从 而 更 有 效 地 提高 整个 系统 的 资源 利用 率 。 例 如 ， 
当 CPU 要 完成 一 组 相关 的 读 ( 或 写 ) 操 作 及 有 关 控 制 时 ,只 需 向 I/O 通道 发 送 一 条 1/O 指 
令 , 以 给 出 其 所 要 执行 的 通道 程序 的 首 地 址 和 要 访问 的 1/0 设备 ,通道 接 到 该 指令 后 ,通过 
执行 通道 程序 便 可 完成 CPU 指定 的 1/0 任务 ,数据 传送 结束 时 向 CPU 发 中 断 请 求 。1/O 
通道 与 DMA 方式 的 区 别 是 : DMA 方式 需要 CPU 来 控制 传输 的 数据 块 大 小 、 传 输 的 内 存 
位 置 ,而 通道 方式 中 这 些 信 息 是 由 通道 控制 的 。 另 外 ,每 个 DMA 控制 器 对 应 一 台 设 备 与 内 
存 传递 数据 ,而 一 个 通道 可 以 控制 多 台 设 备 与 内 存 的 数据 交换 。 


6.3 1/0O 缓冲 


6.3.1 缓冲 的 作用 


由 于 处 理 机 和 外 部 设备 之 间 存 在 巨大 的 速度 差 , 为 了 提高 处 理 机 和 I/O 设备 的 并 行 
性 ,提高 系统 的 效率 ,现代 操作 系统 在 处 理 机 和 LO 设备 交换 数据 的 过 程 中 几乎 全 都 使 用 


了 缓冲 区 来 解决 速度 不 匹配 的 问题 。 可 以 说 ,在 计算 机 系统 中 ,所 有 数据 到 来 和 数据 离开 速 
度 不 匹配 的 地 方 都 使 用 了 缓冲 区 。 缓 冲 主要 有 以 下 几 个 作用 。 

(1) 缓和 CPU 与 1/O 设备 之 间 速 度 不 匹配 的 矛盾 。CPU 和 外 设 之 间 一 般 速 度 差别 很 
大 ,以 打印 机 为 例 ,如 果 没 有 缓冲 区 ,在 打印 机 打印 过 程 中 ,CPU 处 在 忙 等 待 状态 ,浪费 了 大 
量 的 CPU 时间 ; 而 在 CPU 计算 生成 打印 数据 的 过 程 中 ,打印 机 有 处 在 空闲 状态 。 如 果 在 
打印 机 和 CPU 之 间 设 置 缓冲 区 则 可 以 解决 这 个 问题 ,CPU 将 数据 写 入 较 快 的 缓冲 区 后 可 
以 处 理 其 他 任务 ,而 打印 机 可 以 用 较 低 的 速度 实现 打印 。 
(2) 减少 对 CPU 的 中 断 频率 ,放宽 对 CPU 中 断 响应 时 间 的 限制 。 在 远程 数据 通信 中 
使 用 的 一 般 是 串 行 数据 ,也 就 是 按照 位 (bit) 进 行 传输 ,在 计算 机 系统 中 一 般 使 用 的 是 并 行 
数据 ,在 没有 缓冲 区 的 情况 下 ,CPU 需要 每 一 个 位 都 进行 处 理 , 中 断 频 率 高 ,对 中 断 的 响应 
寺 间 要 求 也 高 ,增加 一 个 缓冲 器 可 以 实现 串 行 数据 到 并 行 数据 的 转换 。 这 既 方便 了 计算 机 
对 数据 的 处 理 , 也 降低 了 中 断 频 率 ,放宽 了 对 CPU 中 断 响应 时 间 的 限制 。 
(3) 解决 基本 数据 单元 大 小 ( 即 数据 粒度 ) 不 匹配 的 问题 。 不 同 的 1/O 设备 处 理 数据 的 
基本 单元 也 是 不 同 的 ,比如 磁盘 和 光盘 一 般 是 以 块 为 单位 处 理 数据 ,处 理 机 一 般 是 以 字 为 单 
位 处 理 数据 。 使 用 缓冲 区 可 以 解决 这 个 问题 ,CPU 和 缓冲 区 以 字 为 单位 处 理 数据 ,磁盘 和 
光盘 以 块 为 单位 处 理 数 据 。 

(4) 提高 CPU 和 I/O 设备 之 间 的 并 行 性 。 使 用 缓冲 可 以 大 幅度 提高 CPU 和 1/O 设备 
之 间 的 并 行 性 ,提高 系统 的 吞吐 量 和 设备 的 利用 率 , 如 前 面 打 印 机 例子 所 述 。 

缓冲 根据 位 置 和 数量 的 不 同 可 以 分 为 单 缓冲 、 双 缓冲 、 多 缓冲 和 缓冲 池 , 下 面 分 别 介 绍 。 


6.3.2 单 缓冲 


在 设备 和 处 理 机 之 间 设 置 一 个 缓冲 区 。 设 备 和 处 理 机 交换 数据 时 , 先 把 被 交换 数据 写 
入 缓冲 区 ,然后 需要 数据 的 设备 或 处 理 机 从 缓冲 区 取 走 数据 。 如 图 6-6 所 示 , 在 块 设备 输入 
时 ,假定 从 磁盘 把 一 块 数据 输入 到 缓冲 区 的 时 间 为 工 , 操 作 系统 将 该 缓冲 区 中 的 数据 传送 到 
用 户 区 的 时 间 为 M, 而 CPU 对 这 一 块 数据 处 理 的 时 间 为 C。 由 于 政和 C 是 可 以 并 行 的 , 当 
TC 时 ,系统 对 每 一 块 数据 的 处 理 时 间 为 M 十 T, 反 之 则 为 M 十 C, 故 可 把 系统 对 每 一 块 数 
据 的 处 理 时 间 表 示 为 Max(C,T) 十 M。 

用 户 进程 


人 处理 (C) 
工作 区 | -| 传送 0 组 刘 区 | 输入 DD yo 设备 


图 6-6 单 缓冲 工作 示意 图 


1/O 管理 


媳 口 避 


握 作 系统 原理 


6.3.3 ” 双 缓 冲 


根据 单 缓冲 的 特点 ,CPU 在 传送 时 间 M 内 处 于 空闲 状态 ,由 此 引入 双 缓 冲 。L1/O 设备 


输入 数据 时 先 装填 到 缓冲 区 1, 在 缓冲 区 1 填 满 后 才 开 始 装填 缓冲 
以 从 缓冲 区 1 中 取出 数据 放 入 用 户 进程 处 理 , 当 缓冲 
填 满 , 则 处 理 机 又 从 缓冲 区 2 中 取出 数据 放 入 用 户 进 程 处 理 ,而 1/O 设备 又 可 以 装填 缓冲 区 


1。 双 缓冲 机 制 提高 了 处 理 机 和 


输入 设备 的 并 行 操作 的 程度 。 


区 2, 与 此 同时 处 理 机 可 
区 1 中 的 数据 处 理 完 后 , 若 缓冲 区 2 已 


如 图 6-7 所 示 , 系 统 处 理 一 块 数据 的 时 间 可 以 粗略 地 认为 是 Max(C,T)。 如 果 C=<T， 
可 使 块 设备 连续 输入 (图 中 所 示 情 况 ); 如 果 C 二 T, 则 可 使 CPU 不 必 等 待 设备 输入 。 对 于 
字符 设备 , 若 采用 行 输入 方式 , 则 采用 双 缓 冲 可 使 用 户 在 输入 完 第 一 行 之 后 ,在 CPU 执行 


第 一 行 中 的 命令 的 同时 ,用 户 可 继续 向 第 二 缓冲 区 输入 下 一 行 数据 。 而 单 缓冲 情况 下 则 必 
须 等 待 一 行 数据 被 提取 完毕 才 可 输入 下 一 行 的 数据 。 如 果 两 台 机 器 之 间 通 信 仅 配置 了 单 组 
冲 , 如 图 6-8(a) 所 示 ,那么 ,它们 在 任 一 时 刻 都 只 能 实现 单方 向 的 数据 传输 。 例 如 ,只 允许 把 
数据 从 A 机 传送 到 BB 机 ,或 者 从 B 机 传送 到 人 A 机 ,而 绝 不 允许 双方 同时 向 对 方 发 送 数 据 。 


为 了 实现 双向 数据 传输 ,必须 在 
个 用 作 接 收 缓冲 区 ,如 图 6-8(b) 


用 户 进程 


工作 区 六 = 
Ea 二 


两 台 机 器 中 都 设置 两 个 缓冲 区 ,一 个 用 作 发 送 缓冲 


所 示 。 
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图 6-7 ” 双 缓 冲 工作 示意 图 


上 
Ra 


A 机 B 机 A 机 B 机 
| | | 
| 本 1- 芝 过 :站 性 接收 | 
] 缓冲 区 | | [缓冲 区 
| 缓冲 区 ， | 缓冲 区 ， | 接收 | 发 送 
缓冲 区 | 
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缓冲 区 


(a) 单 缓 六 


(b) 双 缓 促 


图 6-8 双 机 通信 时 缓冲 区 的 设置 


6.3.4 多 缓冲 


区 ; 男 一 


多 缓冲 区 包含 多 个 大 小 相等 的 缓冲 区 ,每 个 缓冲 区 中 有 一 个 链接 指针 指向 下 一 个 缓冲 


区 ,最 后 一 个 缓冲 区 指针 指向 第 一 个 缓冲 区 ,多 个 缓冲 


区 构成 一 个 环 状 。 循 环 缓冲 


于 输 


入 /输出 时 ,还 需要 有 两 个 指针 in 和 out。 对 输入 而 言 ,首先 要 从 设备 接收 数据 到 缓冲 区 中 ， 


in 指针 指向 可 以 输入 数据 的 第 一 个 空 缓冲 区 ; 当 运 行进 程 需要 数据 时 ,从 循环 缓冲 


一 个 装 满 数 据 的 缓冲 区 ,并 从 此 缓冲 区 中 提取 数据 ,out 指针 指向 可 以 提取 数据 的 第 一 个 满 


缓冲 区 。 输 出 则 正好 相反 。 
6.3.5 缓冲 池 


由 多 个 系统 公用 的 缓冲 区 组 成 ,缓冲 区 按 其 使 用 状况 可 以 形成 三 个 队 殉 
装 满 输入 数据 的 缓冲 队列 (输入 队列 ) 和 装 满 输出 数据 的 缓 溃 队列 (输出 队列 
[ 作 缓 冲 区 、 用 于 提取 输入 数据 的 工作 缓冲 区 、 用 于 收容 输 
区 ,如 图 6-9 所 示 。 


种 缓冲 区 : 用 于 收容 输入 数据 的 了 


出 数据 的 工作 缓冲 区 及 用 于 提取 输出 数据 的 工作 缓冲 
缓冲 池 


当 输 入 进程 需要 输入 数据 时 , 便 从 空 缓冲 队列 的 队 首 摘 下 一 个 空 缓冲 


图 6-9 缓冲 区 的 工作 方式 


区 中 取 


: 空 缓冲 队列 、 
)。 还 应 具有 4 


区 ,把 它 作 为 收容 


输入 工作 缓冲 区 ,然后 把 输入 数据 输入 其 中 , 装 满 后 再 将 它 挂 到 输入 队列 队 尾 。 当 计算 进程 


需要 输入 数据 时 , 便 从 输入 队列 取得 一 个 缓冲 


区 作为 提取 输入 了 


[ 作 缓 冲 区 ,计算 进程 从 中 提 


取 数 据 , 数 据 用 完 后 再 将 它 挂 到 空 缓冲 队列 尾 。 当 计算 进程 需要 输出 数据 时 , 便 从 空 缓冲 队 


列 的 队 首 取得 一 个 空 缓冲 区 ,作为 收容 输出 工作 缓冲 


到 输出 队列 队 尾 。 


作为 提取 输出 工作 缓冲 


最 底层 才 涉 及 硬 侍 


区 , 当 其 中 装 满 输出 数据 后 ,再 将 它 挂 


当 要 输出 时 ,由 输出 进程 从 输出 队列 中 取得 一 个 装 满 输 出 数据 的 缓冲 区 ， 


区 , 当 数 据 提取 完 后 ,再 将 它 挂 到 空 缓 冲 队 列 的 队 尾 。 


6.4 独占 设备 的 分 配 


1/O 管理 涉及 的 面 非常 广 , 往 下 与 硬件 有 着 密切 的 联系 , 往 上 又 与 用 户 直接 交互 , 它 与 
进程 管理 ,存储 器 管理 、 文 件 管理 等 都 存在 着 一 定 的 联系 , 即 它们 都 可 能 需要 1/O 软件 来 实 
现 1/O 操作 。 为 了 使 复杂 的 1/O 软件 具有 清晰 的 结构 ,良好 的 可 移植 性 和 适应 性 ,在 I/O 
软件 中 普遍 采用 了 层次 式 结构 ,将 系统 输入 /输出 功能 组 织 成 一 系列 的 层次 ,每 一 层 都 利用 
其 下 层 提供 的 服务 ,完成 输入 /输出 功能 中 的 某 些 子 功能 ， 
并 屏蔽 这 些 功 能 实现 的 细节 ,向 高 层 提供 服务 。 在 层次 式 
结构 的 1/O 软件 中 ,只 要 层次 间 的 接口 不 变 ,对 某 一 层次 
中 的 软件 的 修改 都 不 会 引起 其 下 层 或 高 层 代码 的 变更 , 仅 


F 的 具体 特性 。 


一 个 比较 合理 的 层次 划分 如 图 6-10 所 示 。 整 个 IO 
系统 可 以 看 成 具有 4 个 层次 的 系统 结构 ,各 层次 及 其 功能 


用 户 层 1/O 软 件 


设备 驱动 程序 


设备 独立 性 软件 


中 断 处 理 程序 


硬件 


图 6-10 1/O 系统 的 层次 和 功能 


1/O 管理 


攻 口 避 


所 作 系 统 原 理 


如 下 。 

用 户 层 WO 软件 : 实现 与 用 户 交互 的 接口 ,用 户 可 直接 调用 在 用 户 层 提 供 的 .与 1/O 操 
作 有 关 的 库 函 数 ,对 设备 进行 操作 。 

设备 独立 性 软件 : 用 于 实现 用 户 程序 与 设备 驱动 器 的 统一 接口 .设备 命令 .设备 保护 、 
已 有 设备 分 配 与 释放 等 ,同时 为 设备 管理 和 数据 传送 提供 必要 的 存储 空间 。 

设备 驱动 程序 ; 与 硬件 直接 相关 ,负责 具体 实现 系统 对 设备 发 出 的 操作 指令 ,驱动 IO 
设备 工作 的 驱动 程序 。 
中 断 处 理 程序 : 用 于 保存 被 中 断 进程 的 CPU 环境 , 转 入 相应 的 中 断 处 理 程序 进行 处 
理 , 处 理 完 并 恢复 被 中 断 进程 的 现场 后 ,返回 到 被 中 断 进程 。 

硬件 设备 ; 1/O 设备 通常 包括 一 个 机 械 部 件 和 一 个 电子 部 件 。 为 了 达到 设计 的 模块 性 
和 通用 性 ,一 般 将 其 分 开 : 电子 部 件 称 为 设备 控制 器 (或 适配器 ), 在 个 人 计算 机 中 ,通常 是 
一 块 插入 主板 扩充 槽 的 印刷 电路 板 ; 机 械 部 件 则 是 设备 本 身 。 


6.4.1 设备 的 逻辑 号 和 物理 号 


设备 在 操作 系统 中 一 般 有 两 个 标号 , 即 逻 辑 设备 号 和 物理 设备 号 。 逻 辑 设备 号 用 来 表 
明 设备 的 类 型 。 物 理 设备 号 用 来 表明 具体 的 设备 。 当 用 户 层 软 件 需要 使 用 1/O 设备 时 ,使 
用 逻辑 设备 号 即 可 申请 设备 ,但 是 操作 系统 实际 分 配 设备 时 需要 使 用 设备 的 物理 号 。 为 了 
实现 逻辑 设备 号 到 物理 设备 号 的 映射 ,需要 有 逻辑 设备 表 存 储 二 者 的 对 应 关系 , 迪 辑 设备 表 
的 结构 如 图 6-11 所 示 。 
逻辑 设备 名 物理 设备 名 逻辑 设备 表 的 设置 一 般 有 两 种 方式 : 第 一 种 方式 是 整个 系 
/dev/tty 4 统 中 只 设置 一 个 逻辑 设备 表 , 这 种 方式 的 优点 是 比较 简单 。 缺 点 
ge 3 是 不 同 用 户 不 能 使 用 相同 的 逻辑 设备 名 ,所 以 只 适合 单 用 户 的 系 
2 统 中 。 第 二 种 方式 是 每 个 用 户 一 张 设备 表 。 这 样 可 以 实现 不 同 
图 6-11 逻辑 设备 表 用户 使 用 相同 的 迎 辑 设备 名 ,简化 了 用 户 软 件 的 开发 。 


6.4.2 设备 的 独立 性 


现代 计算 机 系统 常常 配置 了 许多 类 型 的 外 围 设 备 ,同类 设备 又 有 多 台 , 尤 其 是 配置 多 台 
磁盘 机 或 磁带 机 的 情况 很 普遍 。 作 业 在 执行 前 ,应 对 静态 分 配 的 外 围 设备 提出 申请 要 求 , 如 
果 申 请 时 指定 某 一 台 具 体 的 物理 设备 ,那么 分 配 工作 就 很 简单 ,但 当 指定 的 某 台 设备 有 故障 
时 ,就 不 能 满足 申请 ,该 作业 也 就 不 能 投入 运行 。 例 如 ,系统 拥有 A、B 两 台 卡 片 输入 机 , 现 
有 作业 J2 申请 一 台 卡 片 输入 机 ,如 果 它 指定 使 用 A, 那 么 作业 J]l 已 经 占用 A 或 者 设备 A 
坏 了 ,虽然 系统 还 有 同类 设备 B 是 好 的 且 未 被 占用 ,但 也 不 能 接受 作业 J2, 显 然 这 样 做 很 不 
合理 。 为 了 解决 这 一 问题 ,通常 用 户 不 指定 特定 的 设备 ,而 指定 逻辑 设备 ,使 得 用 户 作业 和 
物理 设备 独立 开 来 ,再 通过 其 他 途径 建立 逻辑 设备 和 物理 设备 之 间 的 对 应 关系 ,我 们 称 这 种 
特性 为 “设备 独立 性 ”。 设 备 独 立 性 是 指 操作 系统 把 所 有 外 部 设备 统一 当 作 成 文件 来 看 待 ， 
只 要 安装 它们 的 驱动 程序 ,任何 用 户 都 可 以 像 使 用 文件 一 样 ,操纵 、 使 用 这 些 设 备 , 而 不 必 知 
道 它们 的 具体 存在 形式 。 

具有 设备 独立 性 的 系统 中 ,用 户 编写 程序 时 使 用 的 设备 与 实际 使 用 的 设备 无 关 , 亦 即 逻 
辑 设 备 名 是 用 户 命 名 的 ,可 以 更 改 , 物 理 设备 名 是 系统 规定 的 ,是 不 可 更 改 的。 设备 管理 的 


功能 之 一 就 是 把 逻辑 设备 名 转换 成 物理 设备 名 。 

设备 独立 性 带 来 的 好 处 是 : 用 户 和 物理 的 外 围 设备 无 关 , 系 统 增 减 或 变更 外 围 设备 
时 程序 不 必修 改 ; 易于 处 理 输入 输出 设备 故障 ,例如 , 某 台 行 式 打印 机 发 生 故 障 时 ,可 用 
另 一 台 蔡 换 , 甚 至 可 用 磁带 机 或 磁盘 机 等 不 同类 型 的 设备 代 蔡 ,从 而 提高 了 系统 的 可 靠 
性 ,增加 了 外 围 设备 分 配 的 灵活 性 ,能 更 有 效 地 利用 外 围 设备 资源 ,实现 多 道 程序 设计 
技术 。 

操作 系统 提供 了 设备 独立 特性 后 ,程序 员 可 利用 逻辑 设备 进行 输入 输出 ,而 逻辑 设备 与 
物理 设备 之 间 的 转换 通常 由 操作 系统 的 命令 或 语言 来 实现 。 由 于 操作 系统 大 小 和 功能 不 
同 ,具体 实现 逻辑 设备 到 物理 设备 的 转换 就 有 差别 ,一 般 使 用 以 下 方法 : 利用 作业 控制 语言 
实现 批 处 理 系统 的 设备 转换 ,利用 操作 命令 实现 设备 转换 ,利用 高 级 语言 的 语句 实现 设备 
转换 。 


6.4.3 独占 设备 的 分 配 


在 多 道 程序 环境 下 ,系统 中 的 进程 共享 设备 。 为 防止 进程 对 系统 资源 的 无 序 竞争 ,必须 
系统 统一 分 配 设备 。 系 统 确定 I/O 请 求 的 可 能 性 和 安全 性 后 才 会 按照 分 配 策略 给 用 户 
(进程 ) 分 配 设备 。 在 有 控制 器 和 通道 的 系统 中 ,还 要 考虑 分 配 相 应 的 控制 器 和 通道 。 

为 了 进行 设备 分 配 ,系统 需要 相关 表格 记录 设备 ,控制 器 和 通道 的 相关 信息 。 表 格 有 : 
设备 控制 表 (DCT) ,控制 器 控制 表 (COCT)、 通 道 控制 表 (CHCT) 和 系统 设备 表 (SDT) 等 ， 
下 面 分 别 介绍 。 

(1) 设备 控制 表 (Device Control Table,DCT)。 系 统 为 每 一 个 设备 都 配置 了 一 张 设备 
控制 表 , 用 于 记录 本 设备 的 情况 ; 设备 类 型 type, 设 备 标识 符 deviceid, 设 备 状态 一 一 等 待 / 
不 等 待 , 忙 / 闲 , 指 向 控制 器 表 的 指针 ,重复 执行 次 数 或 时 间 , 设 备 队 列 的 队 首 指针 。 除 此 之 
外 还 应 含有 下 列 字段 。 

设备 队列 队 首 指针 : 凡 因 请 求 本 设备 而 未 得 到 满足 的 进程 ,其 PCB 都 应 按照 一 定 的 策 
略 排 成 一 个 队列 , 称 该 队列 为 设备 请 求 队列 或 设备 队列 。 

设备 状态 : 当 设 备 自身 正 处 于 使 用 状态 时 ,应 将 设备 的 忙 / 闲 标 志 置 “1”。 若 与 该 设备 
相连 接 的 控制 器 或 通道 正 忙 ,也 不 能 启动 该 设备 ,此 时 应 将 设备 的 等 待 标志 置 “1”。 

与 设备 连接 的 控制 器 表 指 针 : 该 指针 指向 与 该 设备 所 连接 的 控制 器 的 控制 表 , 在 设备 
到 主机 之 间 具 有 多 条 通路 的 情况 下 ,一 个 设备 将 与 多 个 控制 器 相连 接 。 此 时 ,在 DCT 中 还 
应 设置 多 个 控制 器 表 指 针 。 
重复 执行 次 数 : 由 于 外 部 设备 在 传送 数据 时 , 较 易 发 生 数 据 传送 错误 ,因而 在 许多 系统 
中 ,如 果 发 生 传送 错误 ,并 不 立即 认为 传送 失败 ,而 是 令 它 重 新 传送 ,并 由 系统 规定 设备 在 工 
作 中 发 生 错误 时 应 重复 执行 的 次 数 。 

(2) 控制 器 控制 表 (Controller Control Table.COCT) 。 系 统 为 每 一 个 控制 器 都 设置 了 
一 张 用 于 记录 本 控制 器 情况 的 控制 器 控制 表 。 其 中 记录 了 : 控制 器 标识 符 (-controllerid) 、 
控制 器 状态 (- 忙 / 闲 ) ,与 控制 器 连接 的 通道 表 指 针 、 控 制 器 队列 的 队 首 指针 、 控 制 器 队列 的 
队 尾 指针 。 

(3) 通道 控制 表 (Channel Control Table.CHCT)。 每 个 通道 都 配 有 一 张 通道 控制 表 。 
其 中 记录 了 : 通道 标识 符 (-channelid)、 通 道 状态 (- 忙 / 闲 ) .与 通道 连接 的 控制 器 表 的 指针 通 
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道 队列 的 队 首 指针 、 通 道 队列 的 队 尾 指针 。 

(4) 系统 统 设 备 表 (System Device Table,SDT)。 这 是 系统 范围 的 数据 结构 ,其 中 记录 
了 系统 中 全 部 设备 的 情况 。 每 个 设备 占 一 个 表 目 ,其 中 包括 设备 类 型 .设备 标识 符 、 设 备 控 
制 表 及 设备 驱动 程序 的 入 口 等 项 。 

在 分 配 设备 时 ,首先 应 考虑 与 设备 分 配 有 关 的 设备 属性 。 设 备 的 固有 属性 可 分 成 三 种 : 
@ 独 占 性 , 指 这 种 设备 在 一 段 时 间 内 只 人 允许 一 个 进程 独占 , 即 “临界 资源 ”; @ 共 享 性 , 指 这 
种 设备 允许 多 个 进程 同时 共享 ; @ 可 虚拟 性 ,设备 本 身 虽 是 独占 设备 ,但 经 过 某 种 技术 处 
理 ,可 以 把 它 改造 成 虚拟 设备 。 

对 于 独占 设备 ,由 于 分 配 不 当 可 能 会 发 生死 锁 , 所 以 需要 先 针 对 是 否 会 发 生死 锁 进 行 安 
全 性 计算 , 仅 当 计 算 结 果 说 明 分 配 是 安全 的 情况 下 才 进 行 设备 分 配 。 在 可 分 配 的 情况 下 按 
如 下 步骤 分 配 。 

分 配 设备 ,首先 根据 I/O 请 求 中 的 物理 设备 名 ,查找 系统 设备 表 (SDT), 从 中 找 出 该 设 
备 的 DCT( 设 备 分 配 表 ), 再 根据 DCT 中 的 设备 状态 字段 ,可 知 该 设备 是 否 正 忙 。 若 忙 , 便 
将 请 求 IO 进程 的 PCB 挂 在 设备 队列 上 ; 否则 , 便 按照 一 定 的 算法 来 计算 本 次 设备 分 配 的 
安全 性 。 如 果 不 会 导致 系统 进入 不 安全 状态 , 便 将 设备 分 配给 请 求 进程 ; 否则 , 仍 将 其 PCB 
插入 设备 等 待 队 列 。 

分 配 控制 器 ,在 系统 把 设备 分 配给 请 求 IO 的 进程 后 ,再 到 其 DCT( 指 向 设备 表 的 指 
针 ) 中 找 出 与 该 设备 连接 的 控制 器 的 COCT( 控 制 器 控制 表 ), 从 COCT 的 状态 字段 中 可 知 
该 控制 器 是 否 忙碌 。 若 忙 , 便 将 请 求 /O 进程 的 PCB 挂 在 该 控制 器 的 等 待 队列 上 ; 否则 ， 
便 将 该 控制 器 分 配给 进程 。 

分 配 通 道 , 通 过 COCT 中 与 控制 器 连接 的 通道 表 指 针 , 找 到 与 该 控制 器 连接 的 通道 的 
CHCT( 通 道 控制 表 ) ,再 根据 CHCT 内 的 状态 信息 ,可 知 该 通道 是 否 忙碌 。 若 忙 , 便 将 请 求 
I/O 的 进程 挂 在 该 通道 的 等 待 队列 上 ; 否则 ,将 该 通道 分 配给 进程 。 只 有 在 设备 .控制 器 和 
通道 三 者 都 分 配 成 功 时 ,这 次 的 设备 分 配 才 算 成 功 。 然 后 , 便 可 启动 该 IO 设备 进行 数据 
传送 。 

仔细 研究 上 述 基本 的 设备 分 配 程序 后 可 以 发 现 ,进程 是 以 物理 设备 名 来 提出 WO 请 
求 的 ; 采用 的 是 单 通路 的 1/0O 系统 结构 ,容易 产生 “瓶颈 ”现象 。 可 以 从 增加 设备 的 独立 
性 和 考虑 都 通路 的 情况 ,以 使 独占 设备 的 分 配 程序 具有 更 强 的 灵活 性 ,并 提高 分 配 的 成 
功率 。 


6.5 设备 处 理 


6.5.1 设备 驱动 程序 


设备 驱动 程序 通常 又 称 为 设备 处 理 程序 ,是 1/O 进程 与 设备 控制 器 之 间 的 通信 程序 ， 
昌 于 常 以 进程 的 形式 存在 ,简称 为 设备 驱动 进程 。 其 主要 任务 是 接收 上 层 软件 发 来 的 抽象 
1/O 要 求 , 转 换 为 具体 要 求 后 ,发 送 给 设备 控制 器 ,启动 设备 去 执行 /O 操作 ; 此 外 ,驱动 程 
序 也 将 从 设备 控制 器 发 来 的 信号 传送 给 上 层 软 件 。 

由 于 驱动 程序 与 硬件 密切 相关 , 故 应 为 每 一 类 设备 配置 一 种 驱动 程序 ,有 时 也 可 为 非常 


类 似 的 两 类 设备 配置 一 个 驱动 程序 。 例 如 ,打印 机 和 显示 器 需要 不 同 的 驱动 程序 ,但 SCSI 
磁盘 驱动 程序 通常 可 以 处 理 不 同 大 小 和 不 同 速度 的 多 个 SCSI 磁盘 ,甚至 还 可 以 处 理 SCSI 
CD-ROM, 

为 了 实现 1/O 进程 与 设备 控制 器 之 间 的 通信 ,设备 驱动 程序 应 具有 以 下 功能 : 接收 由 
设备 独立 性 软件 发 来 的 命令 和 参数 ,并 将 命令 中 的 抽象 要 求 转换 为 具体 要 求 ; 检查 用 户 
1/O 请 求 的 合法 性 ,了 解 1/O 设备 的 状态 ,传递 有 关 参 数 , 设 置 设备 的 工作 方式 ; 发 出 1/O0 
命令 ,如 果 设 备 空闲 , 便 立 即 启动 /O 设备 去 完成 指定 的 I/O 操作 ,如 果 设 备 处 于 忙碌 状 
态 , 则 将 请 求 者 的 请 求 块 挂 在 设备 队列 上 等 待 ; 及 时 响应 由 控制 器 或 通道 发 来 的 中 断 请 求 ， 
并 根据 其 中 断 类 型 调用 相应 的 中 断 处 理 程 序 进行 处 理 ; 对 于 设置 有 通道 的 计算 机 系统 , 驱 
动 程序 还 应 能 够 根据 用 户 的 1/O 请 求 ,自动 地 构成 通道 程序 。 

在 不 同 的 操作 系统 中 所 采用 的 设备 处 理 方式 并 不 完全 相同 。 根 据 在 设备 处 理 时 是 否 设 
置 进程 ,以 及 设置 什么 样 的 进程 而 把 设备 处 理 方式 分 成 以 下 三 类 : @ 为 每 一 类 设备 设置 一 
个 进程 ,专门 用 于 执行 这 类 设备 的 IO 操作 ; 为 所 有 的 交互 式 终端 设置 一 个 交互 式 终端 进 
程 ; 为 同一 类 型 的 打印 机 设置 一 个 打印 进程 。@ 在 整个 系统 中 设置 一 个 1/0 进程 ,专门 用 
于 执行 系统 中 所 有 各 类 设备 的 1/O 操作 。 也 可 以 设置 一 个 输入 进程 和 一 个 输出 进程 ,分 别 
处 理 系 统 中 所 有 各 类 设备 的 输入 和 输出 操作 。@ 不 设置 专门 的 设备 处 理 进程 ,而 只 为 各 类 
设备 设置 相应 的 设备 处 理 程 序 ( 模 块 ) ,供用 户 进程 或 系统 进程 调用 。 

设备 驱动 程序 属于 低级 的 系统 例 程 , 它 与 一 般 的 应 用 程序 及 系统 程序 之 间 有 下 述 明 显 
差异 : 驱动 程序 主要 是 指 在 请 求 /O 的 进程 与 设备 控制 器 之 间 的 一 个 通信 和 转换 程序 。 驱 
动 程序 与 设备 控制 器 和 1/O 设备 的 硬件 特性 紧密 相关 ,因而 对 不 同类 型 的 设备 应 配置 不 同 
的 驱动 程序 。 驱 动 程序 与 I/O 设备 所 采用 的 1/O 控制 方式 紧密 相关 。 了 驱动 程序 与 硬件 紧 
密 相关 ,因而 其 中 的 一 部 分 必须 用 汇编 语言 编写 。 了 驱动 程序 应 允许 可 重 入 。 一 个 正在 运行 
的 驱动 程序 常会 在 一 次 调用 完成 前 被 再 次 调用 。 了 驱动 程序 不 允许 系统 调用 。 

不 同类 型 的 设备 应 有 不 同 的 设备 驱动 程序 ,但 大 体 上 它们 都 可 以 分 成 两 部 分 : 能 够 驱 
动 IO 设备 工作 的 驱动 程序 ; 设备 中 断 处 理 程序 (处 理 1/O 完成 后 的 工作 ) 。 

设备 驱动 程序 的 主要 任务 是 启动 指定 设备 。 启 动 之 前 ,还 必须 完成 必要 的 准备 工作 ,如 
检测 设备 状态 是 否 为 “ 忙 ”等 。 在 完成 所 有 的 准备 工作 后 , 才 向 设备 控制 器 发 送 一 条 启动 命 
令 。 以 下 是 设备 驱动 程序 的 处 理 过 程 : 将 抽象 要 求 转 换 为 具体 要 求 。 由 于 用 户 及 上 层 软 
件 对 设备 控制 器 的 具体 情况 毫 无 了 解 ,因而 只 能 向 它 发 出 抽象 的 要 求 (命令 ) ,但 这 些 命令 无 
法 传送 给 设备 控制 器 。 因 此 就 需要 将 这 些 抽象 要 求 转 换 为 具体 要 求 。 这 一 转换 工作 只 能 由 
驱动 程序 来 完成 ,因为 在 OS 中 只 有 驱动 程序 才 同 时 了 解 抽象 要 求 和 设备 控制 器 中 的 寄存 
器 情况 ; 也 只 有 它 才 知道 命令 .参数 和 数据 应 分 别 送 往 哪 个 寄存 器 。 回 检查 I/O 请 求 的 合 
法 性 。@ 读 出 和 检查 设备 的 状态 。@ 传 送 必 要 的 参数 ,对 于 许多 设备 ,特别 是 块 设备 , 除 必 
须 向 其 控制 嚣 发送 启 动 命令 外 ,还 需 传送 必要 的 参数 。 例 如 ,在 启动 磁盘 进行 读 / 写 之 前 ,应 
先 将 本 次 要 传送 的 字 节 数 和 数据 应 到 达 的 主 存 始 址 , 送 入 控制 器 的 相应 寄存 器 中 。@ 工 作 
方式 的 设置 。 启 动 /O 设备 驱动 程序 发 出 1/O 命令 后 ,基本 的 I/O 操作 是 在 设备 控制 器 的 
控制 下 进行 的 。 通 常 ,1/O 操作 所 要 完成 的 工作 较 多 ,需要 一 定 的 时 间 , 如 读 / 写 一 个 盘 块 中 
的 数据 ,此 时 驱动 (程序 ) 进 程 把 自己 阻塞 起 来 ,直到 中 断 到 来 时 才 将 它 唤醒 。 
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6.5.2 设备 的 中 断 处 理 


1/O 设备 完成 数据 处 理 后 ,由 外 设 通过 接口 电路 向 CPU 发 出 中 断 请 求 信 号 ,CPU 在 满 
足 一 定 的 条 件 下 ,暂停 执行 当前 正在 执行 的 程序 , 转 入 执行 相应 能 够 进行 输入 /输出 操作 的 
程序 , 待 输入 /输出 操作 执行 完毕 之 后 CPU 即 返 回 继 续 执行 原来 被 中 断 的 程序 。 

对 于 外 部 中 断 ,中 断 请 求 信号 是 由 外 部 设备 产生 ,并 施加 到 CPU 的 NMI 或 INTR 引 
脚 上 ,CPU 通过 不 断 地 检测 NMI 和 INTR 引 脚 信号 来 识别 是 否 有 中 断 请 求 发 生 。 对 于 
内 部 中 断 , 中 断 请 求 方式 不 需要 外 部 施加 信号 激发 ,而 是 通过 内 部 中 断 控 制 逻辑 去 调用 。 
无 论 是 外 部 中 断 还 是 内 部 中 断 , 中 断 处 理 过 程 都 要 经 历 以 下 步骤 : 请 求 中 断 一 响应 中 
断 一 关闭 中 断 一 保留 断 点 一 中 断 源 识别 一 保护 现场 一 中 断 服 务 子 程序 一 恢复 现场 一 中 
断 返 回 。 

1. 请 求 中 断 

当 某 一 中 断 源 需 要 CPU 为 其 进行 中 断 服务 时 ,就 输出 中 断 请 求 信号 ,使 中 断 控制 系统 
的 中 断 请 求 触发 器 置 位 ,向 CPU 请 求 中 断 。 系 统 要 求 中 断 请 求 信 号 一 直 保持 到 CPU 对 其 
进行 中 断 响 应 为 止 。 

2. 中 断 响应 

CPU 对 系统 内 部 中 断 源 提出 的 中 断 请 求 必须 响应 ,而 且 自 动 取得 中 断 服务 子 程序 的 入 
口 地 址 ,执行 中 断 服务 子 程序 。 对 于 外 部 中 断 ,CPU 在 执行 当前 指令 的 最 后 一 个 时 钟 周期 
去 查询 INTR 引 脚 , 若 查询 到 中 断 请 求 信号 有 效 , 同 时 在 系统 开 中 断 ( 即 IF 一 1) 的 情况 下 ， 
CPU 向 发 出 中 断 请 求 的 外 设 回 送 一 个 低 电 平 有 效 的 中 断 应 答 信号 ,作为 对 中 断 请 求 INTR 
的 应 答 ,系统 自动 进入 中 断 响应 周期 。 

3. 关闭 中 断 

CPU 响应 中 断后 ,输出 中 断 响应 信号 ,自动 将 状态 标志 寄存 器 FR 或 EFR 的 内 容 压 入 
堆栈 保护 起 来 ,然后 将 FR 或 EFR 中 的 中 断 标 志 位 IF 与 陷阱 标志 位 TF 清 零 ,从 而 自动 关 
闭 外 部 硬件 中 断 。 因 为 CPU 刚 进入 中 断 时 要 保护 现场 ,主要 涉及 堆栈 操作 ,此 时 不 能 再 响 
应 中 断 , 否 则 将 造成 系统 混乱 。 

4. 保护 断 点 

保护 断 点 就 是 将 CS 和 IP/EIP 的 当前 内 容 压 入 堆栈 保存 ,以 便 中 断 处 理 完毕 后 能 返 
被 中 断 的 原 程序 继续 执行 ,这 一 过 程 也 是 由 CPU 自动 完成 的 。 

5. 中 断 源 识别 

当 系统 中 有 多 个 中 断 源 时 ,一 旦 有 中 断 请 求 ,CPU 必须 确定 是 哪 一 个 中 断 源 提出 的 中 
断 请 求 , 并 由 中 断 控制 器 给 出 中 断 服务 子 程序 的 入 口 地 址 , 装 入 CS 与 IP/EIP 两 个 寄存 器 。 
CPU 转 入 相应 的 中 断 服务 子 程序 开始 执行 。 

6. 保护 现场 

主 程序 和 中 断 服务 子 程序 都 要 使 用 CPU 内 部 寄存 器 等 资源 ,为 使 中 断 处 理 程 序 不 破 
坏 主 程序 中 寄存 器 的 内 容 , 应 先 将 断 点 处 各 寄存 器 的 内 容 压 入 堆栈 保护 起 来 ,再 进入 中 断 处 
理 。 现 场 保护 是 由 用 户 使 用 PUSH 指令 来 实现 的 。 

7. 中 断 服务 
中 断 服务 是 执行 中 断 的 主体 部 分 ,不 同 的 中 断 请 求 , 有 各 自 不 同 的 中 断 服务 内 容 , 需 要 


加 


根据 中 断 源 所 要 完成 的 功能 ,事先 编写 相应 的 中 断 服 务 子 程序 存 入 内 存 , 等 待 中 断 请 求 响应 
后 调用 执行 。 

8. 恢复 现场 

当中 断 处 理 完毕 后 ,用 户 通过 POP 指令 将 保存 在 堆栈 中 的 各 个 寄存 器 的 内 容 弹出 , 即 
恢复 主 程序 断 点 处 寄存 器 的 原 值 ,中 断 返 回 。 

在 中 断 服 务 子 程序 的 最 后 要 安排 一 条 中 断 返 回 指令 IRET, 执 行 该 指令 ,系统 自动 将 堆 
栈 内 保存 的 IP/EIP 和 CS 值 弹出 ,从 而 恢复 主 程序 断 点 处 的 地 址 值 ,同时 还 自动 恢复 标志 
寄存 器 FR 或 EFR 的 内 容 ,使 CPU 转 到 被 中 断 的 程序 中 继续 执行 。 


6.6 虚拟 设备 


6.6.1 脱 机 外 围 设备 操作 


脱 机 处 理 是 指 在 不 受 主机 控制 的 外 部 设备 上 进行 数据 处 理 ,或 与 实时 控制 系统 、 主 机 不 
直接 相连 的 数据 处 理 。 常 用 于 主机 速度 不 高 的 数据 处 理 中 提高 设备 的 利用 率 。 

脱 机 处 理 时 ,外 部 设备 上 的 数据 需要 一 个 相当 长 的 等 待 时 间 后 才 被 进行 处 理 。 当 外 部 
设备 上 有 数据 输入 时 ,主机 并 不 予 处 理 , 只 是 将 外 部 设备 的 数据 存放 到 缓冲 区 中 。 一 旦 缓冲 
区 满 了 ,或 是 等 待 的 时 间 到 了 ,主机 才 进 行 加 工 处 理 。 

对 输出 的 操作 也 是 这 样 ,一 旦 计算 机 要 把 处 理 结果 输出 , 它 只 是 把 输出 结果 送 入 缓冲 
中 ,然后 向 外 部 设备 慢 慢 地 进行 输出 ,而 主机 又 去 进行 其 他 的 加 工 处 理 , 当 缓冲 区 中 的 数据 
全 部 输出 完毕 ,主机 再 把 下 一 批 的 数据 存 入 缓冲 区 中 。 

数据 处 理工 作 完全 独立 于 主机 进行 ,使 主机 能 摆脱 慢 速 的 输入 输出 工作 的 牵制 。 如 果 
输入 输出 任务 繁重 , 则 可 配置 若干 台 微型 计算 机 ,一 些 专门 从 事 脱 机 输入 , 另 一 些 专门 从 事 
脱 机 输出 。 

脱 机 处 理 能 提高 设备 的 利用 率 ,但 需要 操作 员 干 预 ,只 适用 于 批 处 理 方式 。 在 一 批 作业 
全 部 输入 到 辅助 存储 器 中 后 ,主机 开始 逐一 处 理 , 这 批 作业 全 部 处 理 完毕 后 才 由 微型 计算 机 
将 它们 分 离 出 来 。 在 一 批 作业 处 理 期 间 , 如 新 来 一 个 作业 ,即使 是 一 个 紧急 任务 且 不 要 花费 
很 多 处 理 时 间 , 系 统 也 不 能 对 它 进 行 处 理 , 只 能 把 它 放 在 下 一 批 中 ,等 待 这 批 作业 处 理 完 后 
再 做 处 理 , 因 而 灵活 性 差 。 


6.6.2 联机 外 围 设备 操作 


与 脱 机 处 理 相对 应 的 是 联机 处 理 , 也 就 是 1/O 处 理 全 部 都 在 主机 的 控制 之 下 执行 。 在 
输入 数据 时 ,如 果 要 对 数据 进行 合法 性 验证 ,就 应 考虑 采用 联机 处 理 的 方式 ,以 便 及 时 发 现 
输入 数据 的 错误 ,并 及 时 予以 更 正 。 联 机 处 理 和 脱 机 处 理 之 间 的 唯一 区 别 由 来 自 服务 的 更 
新 的 频率 决定 ,判定 脱 机 处 理 的 条 件 是 : 在 重新 连接 之 前 不 能 执行 请 求 ,无 论 是 对 信息 的 请 
求 还 是 对 更 新 的 请 求 。 

例如 ,如 果 使 用 的 是 脱 机 电子 邮件 系统 , 则 收 件 箱 可 能 没有 处 于 最 新 状态 ; 不 过 ,这 或 
许 不 能 算是 个 问题 ,因为 电子 邮件 本 身 在 服务 器 端 等 待 被 下 载 时 不 会 发 生 更 改 。 可 是 , 当 脱 
机 使 用 一 个 日 程 编排 系统 时 ,可 能 会 发 现 并 发 问题 ,因为 不 同 的 人 可 能 为 一 个 约会 指定 了 同 
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一 天 的 同一 时 间 。 在 这 种 情况 下 ,系统 必须 能 够 检测 ,并 且 可 能 解决 冲突 。 理 论 上 来 说 ,这 
种 事情 在 联机 时 也 可 能 发 生 ,唯一 的 区 别 是 ,系统 会 更 早 地 通知 预约 冲突 ,可 以 立即 做 出 
反应 。 


6.6.3 SPOOLing 技术 应 用 


在 用 直接 存 取 的 大 容量 磁盘 作为 辅助 存储 器 的 系统 中 ,不 使 用 卫星 机 实现 脱 机 输入 输 
出 ,而 由 主机 和 通道 来 承担 这 一 功能 。 输 入 程序 负责 把 输入 设备 上 的 作业 源源 不 断 地 输入 
到 磁盘 的 某 个 区 域 (作业 输入 区 ) 中 ,并 把 描述 作业 的 信息 登记 在 等 待 队列 中 ,以 供 主机 调 入 
处 理 。 各 作业 要 输出 的 信息 存在 磁盘 的 另 一 个 区 域 ( 作 业 输 出 区 ) 中 , 当 输 出 设备 空闲 时 ,由 
输出 程序 将 输出 区 中 的 信息 输出 。 

由 于 输入 程序 和 输出 程序 的 运行 时 间 很 短 , 仅 仅 是 组 织 信息 的 输入 和 输出 以 及 在 相应 
队列 中 登记 信息 所 需 的 时 间 , 可 使 人 产生 一 种 作业 进入 和 信息 输出 是 脱 机 进行 的 感觉 ,这 称 
为 假 脱 机 输入 输出 系统 。 

这 种 在 联机 情况 下 实现 的 同时 外 围 操 作 称 为 SPOOLing (Simultaneaus Peripheral 
Operating On Line) ,或 称 为 假 脱 机 操作 。 

由 上 述 得 知 ,SPOOLing 技术 是 对 脱 机 输入 、 输 出 系统 的 模拟 。 

SPOOLing 系统 主要 有 以 下 三 部 分 。 

(1) 输入 井 和 输出 井 。 这 是 在 磁盘 上 开辟 的 两 个 大 存储 空间 。 输 入 井 是 模拟 脱 机 输入 
时 的 磁盘 设备 ,用 于 暂 存 1/O 设备 输入 的 数据 ; 输出 井 是 模拟 脱 机 输出 时 的 磁盘 ,用 于 和 暂 存 
用 户 程 序 的 输出 数据 。 

(2) 输入 缓冲 区 和 输出 缓冲 区 。 为 了 缓和 CPU 和 磁盘 之 间 速 度 不 匹配 的 矛盾 ,在 内 存 
中 要 开辟 两 个 缓冲 区 : 输入 缓冲 区 和 输出 缓冲 区 。 输 入 缓冲 区 用 于 暂 存 由 输入 设备 送 来 的 
数据 ,以 后 再 传送 到 输入 井 。 输 出 缓冲 区 用 于 暂 存 从 输出 井 送 来 的 数据 ,以 后 再 传送 给 输出 
设备 。 

(3) 输入 进程 SPi 和 输出 进程 SPo。 这 里 利用 两 个 进程 来 模拟 脱 机 WO 时 的 外 围 控 制 
机 。 其 中 ,进程 SPi 模拟 脱 机 输入 时 的 外 围 控 制 机 ,将 用 户 要 求 的 数据 从 输入 机 通过 输入 组 
冲 区 再 送 到 输入 井 , 当 CPU 需要 输入 数据 时 ,直接 从 输入 井 读 入 内 存 ; 进程 SPo 模拟 脱 机 
输出 时 的 外 围 控 制 机 ,把 用 户 要 求 输出 的 数据 先 从 内 存 送 到 输出 井 , 待 输出 设备 空闲 时 ,再 
将 输出 井中 的 数据 经 过 输出 缓冲 区 送 到 输出 设备 上 。 

SPOOLing 系统 具有 如 下 主要 特点 。 

(1) 提高 了 1/O 的 速度 。 这 里 对 数据 所 进行 的 1/O 操作 ,已 从 对 低速 设备 进行 的 1/O 
操作 ,演变 为 对 输入 井 或 输出 井中 数据 的 存 取 ,如 同 脱 机 输入 输出 一 样 ,提高 了 1/O 速度 ， 
缓和 了 CPU 与 低速 IO 设备 之 间 速 度 不 匹配 的 矛盾 。 

(2) 将 独占 设备 改造 为 共享 设备 。 因 为 在 SPOOLing 系统 中 ,实际 上 没有 为 任何 进程 
分 配 设备 ,而 只 是 在 输入 井 或 输出 井中 为 进程 分 配 一 个 存储 区 和 建立 一 张 /O 请 求 表 。 这 
样 , 便 把 独占 设备 改造 为 共享 设备 。 

(3) 实现 了 虚拟 设备 功能 。 宏 观 上 ,虽然 是 多 个 进程 在 同时 使 用 一 台独 占 设备 ,而 对 于 
每 一 个 进程 而 言 ,它们 都 会 认为 自己 是 独占 了 一 个 设备 。 当 然 ,该 设备 只 是 逻辑 上 的 设备 。 
SPOOLing 系统 实现 了 将 独占 设备 变换 为 若干 台 对 应 的 逻辑 设备 的 功能 


6.7 磁盘 管理 


6.7.1 磁盘 结构 与 性 能 参数 


磁盘 存储 器 ,是 以 磁盘 为 存储 介质 的 存储 器 。 它 是 利用 磁 记 录 技 术 在 涂 有 磁 记 录 介 质 
的 旋转 圆 盘 上 进行 数据 存储 的 辅助 存储 器 ,具有 存储 容量 大 、 数 据 传输 率 高 ,存储 数据 可 长 

在 计算 机 系统 中 ,磁盘 存储 器 常用 于 存放 操作 系统 、 程 序 和 数据 ,是 主 存储 器 的 扩充 。 
发 展 趋势 是 提高 存储 容量 ,提高 数据 传输 率 ,减少 存 取 时 间 , 并 力求 轻薄 、 短 、 小 。 磁 盘存 储 
器 通常 由 磁盘 、 磁 盘 驱动 器 (或 称 磁 盘 机 ) 和 磁盘 控制 器 构成 。 

磁盘 存储 器 是 利用 磁 记 录 技 术 在 旋转 的 圆 盘 介 质 上 进行 数据 存储 的 辅助 存储 器 。 这 是 
一 种 应 用 广泛 的 直接 存 取 存 储 器 。 其 容量 较 主 存储 器 大 千 百 倍 ,在 各 种 规模 的 计算 机 系统 
中 ,常用 作 存 放 操作 系统 .程序 和 数据 ,是 对 主 存储 器 的 扩充 。 磁 盘存 储 器 存 入 的 数据 可 长 
期 保存 ,与 其 他 辅助 存储 器 比较 ,磁盘 存储 器 具有 较 大 的 存储 容量 和 较 快 的 数据 传输 速率 。 
典型 的 磁盘 驱动 器 包括 盘 片 主轴 旋转 机 构 与 驱动 电机 、 头 辟 与 头 壁 支架、 头 辟 驱动 电机 、 净 
化 盘 腔 与 空气 净化 机 构 、 写 入 读 出 电路 \ 伺 服 定位 电路 和 控制 逻辑 电路 等 。 

磁盘 以 恒定 转速 旋转 。 悬 挂 在 头 辟 上 具有 浮动 面 的 头 块 (浮动 磁头 ), 靠 加 载 弹 簧 的 力 
量 压 向 盘面 , 盘 片 表面 带动 的 气流 将 头 块 浮 起 。 头 块 与 盘 片 间 保持 稳定 的 微小 间隙 。 经 滤 
人 尘 器 过 滤 的 空气 不 断送 入 盘 腔 ,保持 盘 片 和 头 块 处 于 高 度 净 化 的 环境 内 ,以 防 头 块 与 盘面 划 
伤 。 根 据 控制 器 送 来 的 磁道 地 址 ( 即 圆 柱 面 地 址 ) 和 寻 道 命令 ,定位 电路 驱动 直线 电机 将 头 
臂 移 至 目标 磁道 上 。 伺 服 磁 头 读 出 伺服 磁道 信号 并 反馈 到 定位 电路 ,使 头 臂 跟 随 伺 服 磁道 
稳定 在 目标 磁道 上 。 读 写 与 选 头 电路 根据 控制 器 送 来 的 磁头 地 址 接 通 应 选 的 磁头 ,将 控制 
器 送 来 的 数据 以 串 行 方式 逐 位 记录 在 目标 磁道 上 ; 或 反之 ,从 选 定 的 磁道 读 出 数据 并 送 往 
控制 器 。 头 臂 装 在 梳 形 架 小 车 上 ,在 寻 道 时 所 有 头 臂 一 同 移动 。 所 有 数据 面 上 相同 直径 的 
同心 圆 磁 道 总 称 圆柱 面 , 即 头 臂 定位 一 次 所 能 存 取 的 全 部 磁道 。 每 个 磁道 都 按 固定 的 格式 
记录 。 在 标志 磁道 起 始 位 置 的 索引 之 后 ,记录 该 道 的 地 址 (圆柱 面 号 和 头号 ) 磁道 的 状况 和 
其 他 参考 信息 。 在 每 一 记录 段 的 尾部 附 记 有 该 段 的 纠 错 码 , 对 连续 少数 几 位 的 永久 缺陷 所 
造成 的 错误 靠 纠 错 码 纠 正 ,对 有 多 位 永久 缺陷 的 磁道 须 用 备份 磁道 代替 。 写 读 操作 是 以 记 
录 段 为 单位 进行 的 。 记 录 段 的 长 度 有 固定 段 长 和 可 变 段 长 两 种 。 

磁盘 的 主要 技术 参数 如 下 。 

平均 潜伏 期 。 这 一 指标 是 指 当 磁头 移动 到 指定 磁道 后 ,要 等 多 长 时 间 指 定 的 读 / 写 扇 区 
会 移动 到 磁头 下 方 ( 盘 片 是 旋转 的 ) , 盘 片 转 得 越 快 ,潜伏 期 越 短 。 平 均 潜 伏 期 是 指 磁盘 转动 
半 圈 所 用 的 时 间 。 显 然 , 同 一 转速 的 硬盘 的 平均 潜伏 期 是 固定 的 。7200rpm 时 约 为 
4.167ms,5400rpm 时 约 为 5. 556ms。 
存储 密度 。 存 储 密度 分 道 密度 ,位 密度 和 面 密 度 。 道 密度 是 沿 磁盘 半径 方向 单位 长 度 
上 的 磁道 数 , 单 位 为 道 /英寸 。 位 密度 是 磁道 单位 长 度 上 能 记录 的 二 进 制 代 码 位 数 ,单位 为 
位 /英寸 。 面 密度 是 位 密度 和 道 密度 的 乘积 ,单位 为 位 /平方 英寸 。 

存储 容量 。 一 个 磁盘 存储 器 所 能 存储 的 字 节 总 数 , 称 为 磁盘 存储 器 的 存储 容量 。 存 储 
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容量 有 格式 化 容量 和 非 格式 化 容量 之 分 ,格式 化 容量 是 指 按照 某 种 特定 的 记录 格式 所 能 
储 信息 的 总 量 , 也 就 是 用 户 可 以 真正 使 用 的 容量 。 非 格式 化 容量 是 磁 记 录 表 面 可 以 利用 的 
磁化 单元 总 数 。 将 磁盘 存储 器 用 于 某 计算 机 系统 中 ,必须 首先 进行 格式 化 操作 ,然后 才能 供 
目 户 记录 信息 。 格 式 化 容量 一 般 是 非 格式 化 容量 的 60% 一 70%。3. 5 英寸 的 硬盘 机 容量 可 
达 4.29GB。 
磁盘 存储 器 平均 存 取 时 间 。 存 取 时 间 是 指 从 发 出 读 写 命令 后 ,磁头 从 某 一 起 始 位 置 移 
动 至 新 的 记录 位 置 ,到 开始 从 盘 片 表面 读 出 或 写 入 信息 所 需要 的 时 间 。 这 段 时 间 由 两 个 数 
值 所 决定 : 一 个 是 将 磁头 定位 至 所 要 求 的 磁道 上 所 需 的 时 间 , 称 为 定位 时 间或 找 道 时 间 ; 
另 一 个 是 找 道 完成 后 至 磁道 上 需要 访问 的 信息 到 达 磁 头 下 的 时 间 , 称 为 等 待 时 间 。 这 两 个 
时 间 都 是 随机 变化 的 ,因此 往往 使 用 平均 值 来 表示 。 平 均 存 取 时 间 等 于 平均 找 道 时 间 与 平 
均等 待 时 间 之 和 。 平 均 找 道 时 间 是 最 大 找 道 时 间 与 最 小 找 道 时 间 的 平均 值 。 平 均 找 道 时 间 
为 10 一 20ms ,平均 等 待 时 间 和 磁盘 转速 有 关 , 它 用 磁盘 旋转 一 周 所 需 时 间 的 一 半 来 表示 , 固 
定 头 盘 转 速 高 达 6000 转 / 分 , 故 平均 等 待 时 间 为 5ms。 

数据 传输 率 。 磁 盘存 储 器 在 单位 时 间 内 向 主机 传送 数据 的 字 节 数 , 叫 数据 传输 率 。 传 
输 率 与 存储 设备 和 主机 接口 逻辑 有 关 。 从 主机 接口 逻辑 考虑 ,应 有 足够 快 的 传送 速度 向 设 
备 接收 /发 送信 息 。 从 存储 设备 考虑 ,假设 磁盘 旋转 速度 为 每 秒 n 转 ,每 条 磁道 容量 为 N 个 
字 节 , 则 数据 传输 率 Dr 一 xzN( 字 节 / 秒 ) ,也 可 以 写成 Dr=D，w( 字 节 / 秒 ), 其 中 ,D 为 位 密 
度 ,v 为 磁盘 旋转 的 线 速度 ,磁盘 存储 器 的 数据 传输 率 可 达 几 十 兆 字 节 / 秒 。 


6.7.2 磁盘 空间 的 管理 


如 6.7.1 节 所 述 ,磁盘 存储 器 按照 扇 区 ,磁道 和 柱 面 进行 划分 。 其 中 , 扇 区 是 磁盘 存储 
器 读 写 的 基本 单位 ,一 般 称 扇 区 为 存储 块 ,对 IO 设备 分 类 时 ,一 般 将 磁盘 存储 设备 称 为 块 
设备 。 格 式 化 过 程 中 会 将 磁盘 的 肩 区 按照 文件 系统 的 格式 进行 组 织 , 实 现 磁 盘存 储 空间 的 
管理 。 


6.7.3 磁盘 调度 策略 


在 多 道 程序 设计 的 计算 机 系统 中 ,各 个 进程 可 能 会 不 断 提出 不 同 的 对 磁盘 进行 读 / 写 操 
作 的 请 求 。 由 于 有 时 候 这 些 进程 的 发 送 请 求 的 速度 比 磁盘 响应 的 还 要 快 ,因此 有 必要 为 每 
个 磁盘 设备 建立 一 个 等 待 队 列 。 

一 次 磁盘 读 写 操作 的 时 间 由 寻找 ( 寻 道 ) 时 间 、 延 迟 时 间 和 传输 时 间 决 定 。 其 中 ,延迟 时 
间 是 随机 值 ,传输 时 间 由 读 写 的 数据 量 决定 ,所 以 磁盘 调度 算法 能 够 直接 决定 的 就 是 寻 道 时 
间 。 通 过 影响 寻 道 时 间 , 磁 盘 调 度 算法 就 影响 了 磁盘 访问 的 总 时 间 。 

常用 磁盘 调度 算法 有 : 先 来 先 服务 算法 、 最 短 寻 找 时 间 优 先 算法 、 扫 描 算法 、 循 环 扫描 
算法 。 

(1) 先 来 先 服务 算法 。FCFS 算法 根据 进程 请 求 访问 磁盘 的 先后 顺序 进行 调度 ,这 是 一 
种 最 简单 的 调度 算法 。 该 算法 的 优点 是 具有 公平 性 。 如 果 只 有 少量 进程 需要 访问 , 且 大 部 
分 请 求 都 是 访问 簇 聚 的 文件 扇 区 , 则 有 望 达到 较 好 的 性 能 ; 但 如 果 有 大 量 进程 竞争 使 用 磁 
盘 , 那 么 这 种 算法 在 性 能 上 往往 接近 于 随机 调度 。 所 以 ,实际 磁盘 调度 中 考虑 一 些 更 为 复杂 
的 调度 算法 。 


算法 思想 : 按 访问 请 求 到 达 的 先后 次 序 服务 。 

优点 : 简单 ,公平 。 

缺点 : 效率 不 高 , 相 邻 两 次 请 求 可 能 会 造成 最 内 到 最 外 的 柱 面 寻 道 , 使 磁头 反复 移动 ， 
增加 了 服务 时 间 , 对 机 械 也 不 利 。 

例 6-1 假设 磁盘 访问 序列 : 98,183,37,122,14,124,65,67。 读 写 头 起 始 位 置 : 53。 
求 : 磁头 服务 序列 和 磁头 移动 总 距离 ( 道 数 ) 。 

由 题 意 和 先 来 先 服务 算法 的 思想 ,得 到 如 图 6-12 所 示 的 磁头 移动 轨迹 。 由 此 : 

磁头 服务 序列 为 : 98,183,37,122,14,124,65,67 

磁头 移动 总 距离 =(98 一 53) 十 (183 一 98) 十 |37 一 183| 十 (122 一 37) 十 |14 一 122| 二 
(124 一 14) 十 165 一 124| 十 (67 一 65) 一 640( 磁 道 ) 


14 37 53 65 67 98 122 124 183 
车 HH [tt | 
2 


图 6-12” 先 来 先 服务 调度 


(2) 最 短 寻找 时 间 优 先 算法 。SSTF 算法 选择 调度 处 理 的 磁道 是 与 当前 磁头 所 在 磁道 
距离 最 近 的 磁道 ,以 使 每 次 的 寻找 时 间 最 短 。 当 然 , 总 是 选择 最 小 寻找 时 间 并 不 能 保证 平均 
寻找 时 间 最 小 ,但 是 能 提供 比 FCFS 算法 更 好 的 性 能 。 这 种 算法 会 产生 “饥饿 ”现象 。 

算法 思想 : 优先 选择 距 当前 磁头 最 近 的 访问 请 求 进行 服务 ,主要 考虑 寻 道 优先 。 

优点 : 改善 了 磁盘 平均 服务 时 间 。 

缺点 : 造成 某 些 访问 请 求 长 期 等 待 得 不 到 服务 。 

例 6-2 对 上 例 的 磁盘 访问 序列 ,可 得 磁头 移动 的 轨迹 如 图 6-13 所 示 。 


14 0 53 65 67 98 122 124 183 


图 6-13 最 短 寻 找 时 间 优 先 算法 
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(3) 扫描 算法 (又 称 电梯 算法 )。SCAN 算法 在 磁头 当前 移动 方向 上 选择 与 当前 磁头 所 
在 磁道 距离 最 近 的 请 求 作为 下 一 次 服务 的 对 象 。 由 于 磁头 移动 规律 与 电梯 运行 相似 , 故 又 
称 为 电梯 调度 算法 。SCAN 算法 对 最 近 扫 描 过 的 区 域 不 公平 ,因此 , 它 在 访问 局 部 性 方面 不 
如 FCFS 算法 和 SSTF 算法 好 。 
算法 思想 : 当 设 备 无 访问 请 求 时 ,磁头 不 动 ; 当 有 访问 请 求 时 ,磁头 按 一 个 方向 移动 ， 
在 移动 过 程 中 对 遇 到 的 访问 请 求 进行 服务 ,然后 判断 该 方向 上 是 否 还 有 访问 请 求 ,如 果 有 则 
继续 扫描 ; 否则 改变 移动 方向 ,并 为 经 过 的 访问 请 求 服务 ,如 此 反复 ,如 图 6-14 所 示 。 
优点 : 克服 了 最 短 寻 道 优先 的 缺点 , 既 考虑 了 距离 , 同 
时 又 考虑 了 方向 。 
循环 扫描 算法 ,在 扫描 算法 的 基础 上 规定 磁头 单 向 移 
动 来 提供 服务 , 回 返 时 直接 快速 移动 至 起 始 端 而 不 服务 任 
何 请 求 。 由 于 SCAN 算法 偏向 于 处 理 那 些 接 近 最 里 或 最 
外 的 磁道 的 访问 请 求 ,所 以 使 用 改进 型 的 C-SCAN 算法 来 
避免 这 个 问题 。 
采用 SCAN 算法 和 C-SCAN 算法 时 磁头 总 是 严格 地 
图 6-14 扫描 算法 (电梯 算法 ) 的 ”遵循 从 盘面 的 一 端 到 另 一 端 ,显然 ,在 实际 使 用 时 还 可 以 
磁头 移动 轨迹 改进 , 即 磁头 移动 只 需要 到 达 最 远 端 的 一 个 请 求 即 可 返 
回 ,不 需要 到 达 磁 盘 端 点 。 这 种 形式 的 SCAN 算法 和 
C-SCAN 算法 称 为 LOOK 和 C-LOOK 调度 。 这 是 因为 它们 在 朝 一 个 给 定 方向 移动 前 会 查 
看 是 否 有 请 求 。 注 意 , 若 无 特别 说 明 . 也 可 以 默认 SCAN 算法 和 C-SCAN 算法 为 LOOK 和 
C-LOOK 调度 。 


6.7.4 RAID 技术 


RAID 是 英文 Redundant Array of Independent Disks 的 缩写 ,中 文 简称 为 独立 元 余 磁 
盘 阵列 。 简 单 地 说 ,RAID 是 一 种 把 多 块 独立 的 硬盘 (物理 硬盘 ) 按 不 同 的 方式 组 合 起 来 形 
成 一 个 硬盘 组 (逻辑 硬盘 ) ,从 而 提供 比 单个 硬盘 更 高 的 存储 性 能 和 提供 数据 备份 技术 。 

在 计算 机 发 展 的 初期 “大 容量 硬盘 的 价格 还 相当 高 ,解决 数据 存储 安全 性 问题 的 主要 
方法 是 使 用 磁带 机 等 设备 进行 备份 ,这 种 方法 虽然 可 以 保证 数据 的 安全 ,但 查阅 和 备份 工作 

1987 年 ,Patterson、Gibson 和 Katz 3 位 工程 师 在 加 州 大 学 伯克利 分 校 发 表 了 题 为 A 
Case of Redundant Array of Inexpensive Disks( 廉 价 磁 盘 宛 余 阵列 方案 ) 的 论文 ,其 基本 
思想 就 是 将 多 只 容量 较 小 的 、 相 对 廉价 的 硬盘 驱动 器 进行 有 机 组 合 ,使 其 性 能 超过 一 只 昂贵 
的 大 硬盘 。 

这 一 设计 思想 很 快 被 接受 ,从 此 RAID 技术 得 到 了 广泛 应 用 ,数据 存储 进入 了 更 快速 、 
更 安全 ,更 廉价 的 新 时 代 。 

磁盘 阵列 对 于 个 人 计算 机 用 户 , 还 是 比较 陌生 和 神秘 的 。 印 象 中 的 磁盘 阵列 似乎 还 停 
留 在 这 样 的 情景 中 :在 宽阔 的 大 厅 里 ,林立 的 磁盘 柜 , 数 名 表情 阴郁 .早早 谢 顶 的 工程 师 徘 徊 
在 其 中 ,不 断 从 中 抽出 一 块 块 沉重 的 硬盘 ,再 插入 一 块 块 似 乎 更 加 沉重 的 硬盘 …… 终于 , 随 
着 大 容量 硬盘 的 价格 不 断 降 低 , 个 人 计算 机 的 性 能 不 断 提升 ,IDE-RAID 作为 磁盘 性 能 改善 


的 最 廉价 解决 方案 ,开始 走 入 一 般 用 户 的 计算 机 系统 。 

RAID 技术 主要 包含 RAID 0~RAID 7 等 数 个 规范 ,它们 的 侧重 点 各 不 相同 ,常见 的 规 
范 有 如 下 几 种 。 

RAID 0: 连续 以 位 或 字 节 为 单位 分 割 数据 ,并 行 读 / 写 于 多 个 磁盘 上 ,因此 具有 很 高 的 
数据 传输 率 , 但 它 没有 数据 元 余 , 因 此 并 不 能 算是 真正 的 RAID 结构 。RAID 0 只 是 单纯 地 
提高 性 能 ,并 没有 为 数据 的 可 靠 性 提供 保证 ,而 且 其 中 的 一 个 磁盘 失效 将 影响 到 所 有 数据 。 
因此 ,RAID 0 不 能 应 用 于 数据 安全 性 要 求 高 的 场合 。 

RAID 1: 它 是 通过 磁盘 数据 镜像 实现 数据 元 余 , 在 成 对 的 独立 磁盘 上 产生 互 为 备份 的 
数据 。 当 原始 数据 繁忙 时 ,可 直接 从 镜像 备份 中 读 取 数 据 , 因 此 RAID 1 可 以 提高 读 取 性 
能 。RAID 1 是 磁盘 阵列 中 单位 成 本 最 高 的 ,但 提供 了 很 高 的 数据 安全 性 和 可 用 性 。 当 一 
个 磁盘 失效 时 ,系统 可 以 自动 切换 到 镜像 磁盘 上 读 写 , 而 不 需要 重组 失效 的 数据 。 

RAID 0 十 1: 也 被 称 为 RAID 10 标准 ,实际 是 将 RAID 0 和 RAID 1 标准 结合 的 产物 ， 
在 连续 地 以 位 或 字 节 为 单位 分 割 数据 并 且 并 行 读 / 写 多 个 磁盘 的 同时 ,为 每 一 块 磁盘 做 磁盘 
镜像 进行 元 余 。 它 的 优点 是 同时 拥有 了 RAID 0 的 超凡 速度 和 RAID 1 的 数据 高 可 靠 性 ,但 是 
CPU 占用 率 同样 也 更 高 ,而 且 磁盘 的 利用 率 比较 低 。 

RAID 2: 将 数据 条 块 化 地 分 布 于 不 同 的 硬盘 上 ,条 块 单位 为 位 或 字 节 ,并 使 用 称 为 “加 
重 平均 纠 错 码 ( 海 明码 )” 的 编码 技术 来 提供 错误 检查 及 恢复 。 这 种 编码 技术 需要 多 个 磁盘 
存放 检查 及 恢复 信息 ,使 得 RAID 2 技术 实施 更 复杂 ,因此 在 商业 环境 中 很 少 使 用 。 

RAID 3: 它 同 RAID 2 非常 类 似 , 都 是 将 数据 条 块 化 分 布 于 不 同 的 硬盘 上 ,区 别 在 于 
RAID 3 使 用 简单 的 奇偶 校 验 ,并 用 单 块 磁盘 存放 奇偶 校 验 信 息 。 如 果 一 块 磁盘 失效 ,奇偶 
盘 及 其 他 数据 盘 可 以 重新 产生 数据 ;如 果 奇 偶 盘 失效 则 不 影响 数据 使 用 。RAID 3 对 于 大 
量 的 连续 数据 可 提供 很 好 的 传输 率 , 但 对 于 随机 数据 来 说 ,奇偶 盘 会 成 为 写 操作 的 瓶颈 。 

RAID 4: RAID 4 同样 也 将 数据 条 块 化 并 分 布 于 不 同 的 磁盘 上 ,但 条 块 单位 为 块 或 记 
录 。RAID 4 使 用 一 块 磁盘 作为 奇偶 校 验 盘 ,每 次 写 操作 都 需要 访问 奇偶 盘 , 这 时 奇偶 校 验 
盘 会 成 为 写 操作 的 瓶颈 ,因此 RAID 4 在 商业 环境 中 也 很 少 使 用 。 

RAID 5: RAID 5 不 单独 指定 奇偶 盘 , 而 是 在 所 有 磁盘 上 交叉 地 存 取 数 据 及 奇偶 校 验 
信息 。 在 RAID 5 上 , 读 / 写 指针 可 同时 对 阵列 设备 进行 操作 ,提供 了 更 高 的 数据 流量 。 
RAID 5 更 适合 于 小 数据 块 和 随机 读 写 的 数据 。 

RAID 3 与 RAID 5 相 比 ,最 主要 的 区 别 在 于 RAID 3 每 进行 一 次 数据 传输 就 需 涉及 所 
有 的 阵列 盘 ; 而 对 于 RAID 5 来 说 ,大 部 分 数据 传输 只 对 一 块 磁 盘 操 作 , 并 可 进行 并 行 操作 。 
在 RAID 5 中 有 “ 写 损失 ”, 即 每 一 次 写 操 作 将 产生 4 个 实际 的 读 / 写 操作 ,其 中 两 次 读 旧 的 
数据 及 奇偶 信息 ,两 次 写 新 的 数据 及 奇偶 信息 。 

RAID 6: 与 RAID 5 相 比 ,RAID 6 增加 了 第 二 个 独立 的 奇偶 校 验 信息 块 。 两 个 独立 的 
奇偶 系统 使 用 不 同 的 算法 ,数据 的 可 靠 性 非常 高 ,即使 两 块 磁盘 同时 失效 也 不 会 影响 数据 的 
使 用 。 但 RAID 6 需要 分 配给 奇偶 校 验 信息 更 大 的 磁盘 空间 ,相对 于 RAID 5 有 更 大 的 “ 写 
损失 ”, 因 此 “ 写 性 能 ”非常 差 。 较 差 的 性 能 和 复杂 的 实施 方式 使 得 RAID 6 很 少 得 到 实际 
应 用 。 

RAID 7: 这 是 一 种 新 的 RAID 标准 ,其 自身 带 有 智能 化 实时 操作 系统 和 用 于 存储 管理 
的 软件 工具 ,可 完全 独立 于 主机 运行 ,不 占用 主机 CPU 资源 。RAID 7 可 以 看 作 是 一 种 存 
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储 计算 机 (Storage Computer), 它 与 其 他 RAID 标准 有 明显 区 别 。 除 了 以 上 的 各 种 标准 ,我 
们 可 以 如 RAID 0 十 1 那样 结合 多 种 RAID 规范 来 构筑 所 需 的 RAID 阵列 ,例如 ,RAID 5 十 3 
(RAID 53) 就 是 一 种 应 用 较为 广泛 的 阵列 形式 。 用 户 一 般 可 以 通过 灵活 配置 磁盘 阵列 来 获 
得 更 加 符合 其 要 求 的 磁盘 存储 系统 。 


6.8 磁盘 高 速 缓存 


磁盘 缓存 ,又 称 磁盘 快 取 , 实 际 上 就 是 将 下 载 到 的 数据 先 保存 于 系统 为 软件 分 配 的 内 存 
空间 中 (这 个 内 存 空间 被 称 为 “内 存 池 ”) , 当 保存 到 内 存 池 中 的 数据 达到 一 个 程度 时 , 便 将 数 
据 保存 到 硬盘 中 。 这 样 可 以 减少 实际 的 磁盘 操作 ,有 效 地 保护 磁盘 免 于 重复 的 读 写 操作 而 
导致 的 损坏 。 

磁盘 缓存 是 为 了 减少 CPU 透 过 I/O 读 取 磁 盘 机 的 次 数 , 提 升 磁盘 1/O 的 效率 ,用 一 块 
内 存 来 储存 存 取 较 频繁 的 磁盘 内 容 ; 因为 内 存 的 存 取 是 电子 动作 ,而 磁盘 的 存 取 是 1/0O 动 
作 , 感 觉 上 磁盘 IO 变 得 较为 快速 。 相 同 的 技巧 可 用 在 写 入 动作 ,我 们 先 将 欲 写 入 的 内 容 
放 入 内 存 中 ,等 到 系统 有 其 他 空闲 的 时 间 , 再 将 这 块 内 存 的 资料 写 入 磁盘 中 。 


6.8.1 设计 考虑 因素 


设计 磁盘 高 速 缓存 ,主要 考虑 如 下 因素 。 

首先 , 当 一 个 1/0O 请 求 从 一 个 磁盘 高 速 缓存 中 得 到 满足 时 ,磁盘 高 速 缓存 中 的 数据 必 
须 传送 到 发 送 请 求 的 进程 。 这 可 以 通过 在 内 存 中 把 这 一 块 数据 从 磁盘 高 速 缓存 中 传送 到 分 
配给 用 户 进程 的 存储 空间 中 ,或 者 简单 地 通过 使 用 一 个 共享 内 存 , 传 送 指向 磁盘 高 速 缓存 中 
相应 的 指针 。 

第 二 个 因素 就 是 高 速 缓存 的 置换 策略 。 常 用 的 置换 策略 有 以 下 几 种 。 

(1) 先进 先 出 算法 , 即 FIFO 算法 (First-In First-Out algorithm)。 这 种 算法 选择 最 先 
调 入 缓存 的 块 作为 被 替换 的 页 面 。 它 的 优点 是 比较 容易 实现 ,能 够 利用 缓存 中 块 调度 情况 
的 历史 信息 ,但 是 ,没有 反映 程序 的 局 部 性 。 因 为 最 先 调 入 缓存 的 块 ,很 可 能 也 是 经 常 要 使 
的 页 面 。 

(2) 近期 最 少 使 用 算法 , 即 LFU 算法 (Least Frequently Used algorithm) 。 这 种 算法 选 
择 近期 最 少 访问 的 块 作为 被 蔡 换 的 块 。 显 然 ,这 是 一 种 非常 合理 的 算法 ,因为 到 目前 为 止 最 
少 使 用 的 块 ,很 可 能 也 是 将 来 最 少 访问 的 块 。 该 算法 既 充分 利用 了 缓存 中 块 调度 情况 的 历 
史 信 息 , 又 正确 反映 了 程序 的 局 部 性 。 但 是 ,这 种 算法 实现 起 来 非常 困难 。 

(3) 最 久 没有 使 用 算法 , 即 LRU 算法 (Least Recently Used algorithm) 。 这 种 算法 把 
近期 最 久 没有 被 访问 过 的 块 作为 被 替换 的 块 。 它 把 LFU 算法 中 要 记录 数量 上 的 “多 ”与 
“ 少 ” 简 化 成 判断 * 有” 与“ 无”, 因此 ,实现 起 来 比较 容易 。 


6.8.2 性 能 考虑 因素 


磁盘 高 速 缓存 的 性 能 问题 可 以 简化 为 是 否 可 以 达到 特定 的 命中 率 。 这 取决 于 访问 磁盘 
的 局 部 性 行为 .置换 算法 和 其 他 设计 因素 。 一 般 地 ,命中 率 主要 是 磁盘 高 速 缓存 大 小 的 函 
数 。 访 问 模式 的 顺序 和 相关 的 设计 问题 ,如 块 大 小 ,将 对 性 能 产生 重要 影响 。 


6.9 磁盘 讨论 
6.9.1 固态 盘 


固态 硬盘 (Solid State Drives) ,简称 固 盘 ,是 用 固态 电子 存储 芯片 阵列 而 制 成 的 硬盘 ， 
日 控制 单元 和 存储 单元 (Flash 芯片 .DRAM 芯片 ) 组 成 。 固 态 硬 盘 在 接口 的 规范 和 定义 、 功 
能 及 使 用 方法 上 与 普通 硬盘 的 完全 相同 ,在 产品 外 形 和 尺寸 上 也 完全 与 普通 硬盘 一 致 ,被 广 
泛 应 用 于 军事 .车 载 .工控 .视频 监控 ,网 络 监 控 , 网 络 终端 .电力 医疗 .航空 .导航 设备 等 
领域 。 
固态 硬盘 的 存储 介质 分 为 两 种 ,一 种 是 采用 闪存 (Flash 芯片 ) 作 为 存储 介质 ,另外 一 种 
是 采用 DRAM 作为 存储 介质 。 

基于 闪存 类 : 基于 闪存 的 固态 硬盘 (IDEFlash Disk Serial ATA Flash Disk) : 采用 
Flash 芯片 作为 存储 介质 ,这 也 是 通常 所 说 的 SSD。 它 的 外 观 可 以 被 制作 成 多 种 模样 , 例 
如 ,笔记 本 硬盘 、 微 硬盘 、 存 储 卡 、U 盘 等 样式 。 这 种 SSD 固态 硬盘 最 大 的 优点 就 是 可 以 移 
动 , 而 且 数据 保护 不 受 电源 控制 ,能 适应 于 各 种 环境 ,适合 于 个 人 用 户 使 用 。 

基于 DRAM 类 : 基于 DRAM 的 固态 硬盘 采用 DRAM 作为 存储 介质 ,应 用 范围 较 窄 。 
它 仿 效 传统 硬盘 的 设计 ,可 被 绝 大 部 分 操作 系统 的 文件 系统 工具 进行 卷 设置 和 管理 ,并 提供 
工业 标准 的 PCI 和 FC 接口 用 于 连接 主机 或 者 服务 器 。 应 用 方式 可 分 为 SSD 硬盘 和 SSD 
硬盘 阵列 两 种 。 它 是 一 种 高 性 能 的 存储 器 ,而 且 使 用 寿命 很 长 ,美中不足 的 是 需要 独立 电源 
来 保护 数据 安全 。DRAM 固态 硬盘 属于 比较 非 主流 的 设备 。 
固态 硬盘 具有 如 下 优点 。 

读 写 速度 快 。 采 用 闪存 作为 存储 介质 , 读 取 速 度 相对 机 械 硬 盘 更 快 。 固 态 硬 盘 不 用 磁 
头 , 寻 道 时 间 几 乎 为 0。 持 续 写 入 的 速度 非常 惊人 ,随机 读 写 速度 快 远 超 机 械 硬 盘 。 最 常见 
的 7200 转机 械 硬盘 的 寻 道 时 间 一 般 为 12 一 14ms, 而 固态 硬盘 可 以 轻易 达到 0. lms 甚至 
更 低 。 

防震 抗 摔 性 。 传 统 硬盘 都 是 磁 碟 型 的 ,数据 储存 在 磁 碟 扇 区 里 。 而 固态 硬盘 是 使 用 闪 
存 颗粒 ( 即 MP3、U 盘 等 存储 介质 ?制作 而 成 ,所 以 SSD 固态 硬盘 内 部 不 存在 任何 机 械 部 
件 , 这 样 即使 在 高 速 移动 甚至 伴随 翻转 倾斜 的 情况 下 也 不 会 影响 到 正常 使 用 ,而 且 在 发 生 碰 
撞 和 震荡 时 能 够 将 数据 丢失 的 可 能 性 降 到 最 小 。 相 较 传 统 硬盘 ,固态 硬盘 占有 绝对 优势 。 

低 功 耗 。 固 态 硬盘 的 功 耗 上 要 低 于 传统 硬盘 。 

无 噪声 。 固 态 硬盘 没有 机 械 马 达 和 风扇 ,工作 时 噪声 值 为 0dB。 基 于 闪存 的 固态 硬盘 
在 工作 状态 下 能 耗 和 发 热量 较 低 (但 高 端 或 大 容量 产品 能 耗 会 较 高 ) 。 内 部 不 存在 任何 机 械 
活动 部 件 , 不 会 发 生机 械 故 障 ,也 不 怕 碰 撞 、 冲 击 、 震 动 。 由 于 固态 硬盘 采用 无 机 械 部 件 的 闪 
存 芯片 ,所 以 具有 发 热量 小 .散热 快 等 特点 。 
工作 温度 范围 大 。 典 型 的 硬盘 驱动 器 只 能 在 5 一 55C 范围 内 工作 。 而 大 多 数 固态 硬盘 
可 在 一 10~70C 工 作 。 固 态 硬盘 比 同 容量 机 械 硬盘 体积 小 .重量 轻 。 固 态 硬盘 的 接口 规范 
和 定义 、 功 能 及 使 用 方法 上 与 普通 硬盘 的 相同 ,在 产品 外 形 和 尺寸 上 也 与 普通 硬盘 一 致 。 其 
芯片 的 工作 温度 范围 很 宽 ( 一 40 一 85YC) 。 
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轻便 。 固 态 硬盘 在 重量 方面 更 轻 , 与 常规 1. 8 英寸 硬盘 相 比 ,重量 轻 20 一 30g。 

除了 具有 上 述 优点 ,固态 硬盘 也 有 如 下 缺点 。 

容量 。 固 态 硬盘 最 大 容量 仅 为 4TB, 由 闪 迪 (SanDisk) 发 布 的 Optimus MAX (和 擎 
天 柱 ) 。 

寿命 限制 。 固 态 硬盘 闪存 具有 擦 写 次 数 限制 的 问题 ,这 也 是 许多 人 诉 病 其 寿命 短 的 原 
因 所 在 。 

售 价 高 。 市 场 上 的 128GB 固态 硬盘 产品 的 价格 大 约 550 元 人 民 币 ,而 256GB 的 产品 
价格 大 约 950 元 人 民 币 (2014 年 价格 ) 。 计 算 下 来 ,每 GB 价格 在 4.2 元 人 民 币 (2014 年 价 
格 ) 左 右 , 依 然 比 传统 机 械 硬 盘 每 GB 0. 3 元 人 民 币 (2014 年 价格 ?的 价格 高 出 了 十 几 倍 。 


6.9.2 智能 磁盘 系统 


本 章 讨 论 的 磁盘 本 身 提供 的 仅仅 是 存储 容量 ,而 这 个 存储 容量 需要 由 操作 系统 来 管理 。 
除了 非常 细微 或 程度 很 轻 的 读 写 错误 外 ,这 种 磁盘 一 般 不 能 检测 和 恢复 本 身 的 错误 。 一 旦 
磁盘 朋 省 ,整个 系统 就 宣告 结束 。 另 一 种 磁盘 是 智能 磁盘 ,这 种 磁盘 具有 一 定 的 智能 ,除了 
提供 存储 容量 外 ,还 能 够 进行 自我 检测 并 从 某 些 错误 中 恢复 。 当 错误 的 频率 和 烈度 达到 一 
定 的 临界 点 时 ,这 种 磁盘 能 够 向 系统 发 出 警报 ,以 便 系 统管 理 员 及 时 更 换 磁 盘 , 避 免 由 磁盘 
衣 省 所 造成 的 信息 丢失 ,这 种 功能 需要 操作 系统 提供 支持 。 

由 于 大 部 分 桌面 操作 系统 不 支持 在 线 磁 盘 更换 , 这 种 在 线 更 换 的 能 力 通常 由 所 谓 的 智 
能 存储 系统 提供 。 智 能 存储 系统 是 存储 设备 里 面 的 顶级 大 师 , 通 常 由 许多 智能 和 非 智 能 的 
磁盘 构成 ,存储 容量 一 般 在 TB 级 ,现在 更 常 在 PB 级 ,所 以 智能 存储 系统 有 时 也 称 作 海量 存 
储 设备 。 这 类 存储 设备 有 自己 的 中 央 处 理 器 和 高 速 缓存 ,有 自己 的 操作 系统 ,能 够 独立 于 服 
务 器 而 存在 和 运行 。 


小 结 


现代 计算 机 系统 都 连接 了 许多 不 同类 型 的 1/O 设备 ,不 同类 型 的 1/0 设备 具有 不 同 的 
速度 和 访问 特性 ,使 用 缓冲 区 可 以 较 好 地 平衡 处 理 机 和 1/O 设备 之 间 的 速度 差异 。 

软件 系统 开发 应 该 独立 于 具体 的 硬件 设备 ,这 样 才能 达到 好 的 通用 性 和 移植 性 ,使 用 分 
层 的 体系 结构 可 以 将 用 户 程 序 和 具体 硬件 隔离 开 , 不 同 层 完成 不 同 的 任务 。 使 用 驱动 程序 
结构 有 利于 系统 添加 新 的 硬件 ,给 系统 的 扩充 提供 便利 。 

磁盘 系统 能 够 存储 大 量 信息 。 传 统 的 机 械 磁 盘 系 统 具 有 特殊 的 硬件 结构 ,使 用 合适 的 
调度 算法 可 以 提高 整体 的 数据 传输 速度 。RAID 系统 使 用 价格 较 低 的 多 个 磁盘 组 成 了 大 容 
量 稳定 的 磁盘 ,可 以 很 好 地 容错 。 固 态 硬 盘 提 供 了 更 好 的 便携 性 。 
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计算 机 系统 在 运行 过 程 中 需要 使 用 程序 来 处 理 大 量 的 数据 。 程 序 和 数据 需要 在 内 存 中 
才能 被 计算 机 处 理 。 但 是 内 存 的 容量 有 限 , 而 且 是 易 失 存储 器 ,可 以 将 程序 和 数据 存储 在 外 
部 存储 器 中 。 外 部 存储 器 中 的 存储 内 容 以 文件 的 形式 被 组 织 在 一 起 ,为 了 方便 用 户 使 用 并 
提高 安全 性 和 并 发 情况 下 的 一 致 性 ,文件 由 操作 系统 进行 统一 管理 。 这 就 需要 操作 系统 提 
供 文件 管理 功能 ,文件 管理 功能 包括 文件 的 存 取 、 共 享 和 保护 等 。 


7.0 问题 导入 


在 现代 计算 机 系统 中 ,要 用 到 大 量 的 程序 和 数据 ,由 于 内 存 容量 有 限 且 无 法 长 期 保存 ， 
所 以 它们 是 以 文件 的 形式 存放 在 外 存 ,需要 时 再 将 它们 调 入 内 存 。 用 户 和 系统 要 频繁 地 对 
它们 进行 访问 ,如 果 让 用 户 在 程序 中 自己 安排 它们 在 外 存 的 存放 位 置 ,不 仅 要 求 用 户 熟 悉 外 
存 的 特性 、 各 种 文件 的 属性 以 及 它们 在 外 存 上 的 位 置 ,而 且 在 多 用 户 环 境 下 ,还 必须 保证 数 
据 的 安全 性 和 一 致 性 。 这 些 工作 是 用 户 能 胜任 的 吗 ? 


7.1 文件 管理 概述 


7.1.1 文件 和 文件 系统 


文件 (File) 是 操作 系统 中 的 一 个 重要 概念 。 文 件 可 以 有 如 下 定义 : @ 文 件 是 软件 机 构 、 
软件 资源 的 管理 方式 ; @@ 具 有 符号 名 的 一 组 相关 元 素 的 有 序 序列 ,是 一 段 程序 或 数据 的 集 
合 ; 加 一 组 赋 名 的 相关 联 字符 流 的 集合 ,或 者 是 相关 记录 。 而 记录 是 有 意义 的 信息 集合 。 
在 系统 运行 时 ,计算 机 以 进程 为 基本 单位 进行 资源 的 调度 和 分 配 ; 而 在 用 户 进行 的 输 
入 、 输 出 中 , 则 以 文件 为 基本 单位 。 大 多 数 应 用 程序 的 输入 都 是 通过 文件 来 实现 的 ,其 输出 
也 都 保存 在 文件 中 ,以 便 信 息 的 长 期 存储 及 将 来 的 访问 。 当 用 户 将 文件 用 于 应 用 程序 的 输 
入 、 输 出 时 ,还 希望 可 以 访问 文件 、 修 改 文 件 和 保存 文件 等 ,实现 对 文件 的 维护 管理 ,这 就 需 
要 系统 提供 一 个 文件 管理 系统 ,操作 系统 中 的 文件 系统 (File System) 就 是 用 于 实现 用 户 的 
这 些 管理 要 求 。 

从 用 户 的 角度 看 ,文件 系统 是 操作 系统 的 重要 部 分 之 一 。 用 户 关 心 的 是 如 何 命名 、 分 类 
和 查找 文件 ,如 何 保证 文件 数据 的 安全 性 以 及 对 文件 可 以 进行 哪些 操作 等 。 而 对 其 中 的 细 
节 , 如 文件 如 何 存 储 在 辅 存 上 、 如 何 管理 辅 存 等 问题 很 少 关心 。 

文件 系统 提供 了 与 二 级 存储 相关 的 资源 的 抽象 ,让 用 户 能 在 不 了 解 文件 的 各 种 属性 、 文 
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件 存储 介质 的 特征 以 及 文件 在 存储 介质 上 的 具体 位 置 等 情况 下 ,方便 快捷 地 使 用 文件 。 用 
户 通过 文件 系统 建立 文件 ,提供 应 用 程序 的 输入 、 输 出 ,对 资源 进行 管理 。 


7.1.2 文件 的 分 类 和 结构 


1. 文件 的 分 类 

系统 中 的 文件 具有 多 种 类 型 ,文件 的 分 类 方法 主要 有 以 下 几 种 方法 。 

1) 按 文件 性 质 和 用 途 分 类 

系统 文件 : 由 系统 软件 构成 的 文件 。 只 人 允许 用 户 通过 系统 调用 或 系统 提供 的 专用 命令 
来 执行 它们 ,不 允许 对 其 进行 读 写 和 修改 。 这 些 文件 主要 由 操作 系统 核心 和 各 种 系统 应 用 
程序 或 实用 工具 程序 和 数据 组 成 。 例 如 ,msdos. sys、io. sys 及 UNIX 系统 下 的 核心 文件 
/unix。 

库 文 件 : 这 类 文件 允许 用 户 对 其 进行 读 取 和 执行 ,但 不 允许 对 其 进行 修改 ,主要 由 各 种 
标准 子 程 序 库 组 成 。 例 如 ,C 语言 的 * .LIB、UNIX 系统 下 的 /lib、/usr/lib 目录 下 的 文件 。 

用 户 文件 : 这 类 文件 是 用 户 通过 操作 系统 保存 的 用 户 文件 ,由 文件 的 所 有 者 或 所 有 者 
授权 的 用 户 才 能 使 用 ,用 户 将 这 些 文件 委托 和 操作 系统 保管 。 主 要 由 用 户 的 源 程序 、 可 执行 
目标 程序 .用户 数 据 库 组 成 。 例 如 ,x* .c、* .dbf、x*.o 等 。 

2) 按 操作 保护 分 类 

只 读 文件 (Read Only) : 只 允许 文件 主 及 被 核准 的 用 户 去 读 文件 ,而 不 允许 写 文件 。 

可 读 写 文件 (Read/ Write) : 允许 文件 主 及 被 核准 的 用 户 去 读 和 写 文件 。 

可 执行 文件 (Execute) : 允许 文件 主 及 被 核准 的 用 户 去 调用 执行 文件 而 不 允许 读 和 写 
fe 

各 个 操作 系统 的 保护 方法 和 级 别 有 所 不 同 。 

DOS 操作 系统 的 文件 保护 有 三 种 : 系统 (System,S)、 隐 藏 (Hide, H)、 可 写 (Write， 
W)。UNIX 操作 系统 的 文件 保护 有 9 种 ,分 为 三 组 ,分 别 为 文件 主 (Owner) 、 同 组 (Group)、 
其 他 (Other)。 每 组 均 有 r、w、x 的 权限 控制 。 

3) 按 使 用 情况 分 类 
临时 文件 (Temporary File) :用 于 系统 在 工作 过 程 中 产生 的 中 间 文 件 , 一 般 有 暂 存 的 目 
录 , 如 \temp /tmp /temporary file。 正 常情 况 下 ,工作 完毕 后 会 自动 删除 ,异常 中 断 时 可 能 
会 残留 一 些 临 时 文件 。 
永久 文件 : 指 受 系统 管理 的 各 种 系统 文件 和 用 户 文件 ,经 过 安装 、 编 辑 ` 编 译 生 成 的 文 
件 , 存 放 在 硬盘 .光盘 等 外 部 设备 上 。 
档案 文件 : 系统 或 一 些 使 用 工具 软件 包 在 工作 过 程 中 记录 在 案 的 文档 资料 文件 ,以 便 
查阅 历史 档案 。 例 如 , * .hst\*.log、*.CHK 等 。 

4) 按 用 户 观点 分 类 (UNIX 或 Linux 操作 系统 ) 

普通 文件 (常规 文件 ) : 是 系统 中 最 一 般 组 织 格式 的 文件 ,包含 系统 文件 ,用户 文件 和 库 
函数 文件 .实用 程序 文件 等 。 

目录 文件 : 是 由 文件 的 目录 信息 构成 的 特殊 文件 ,操作 系统 将 目录 也 称 为 文件 ,便于 统 
一 管理 。 


特殊 文件 (设备 驱动 程序 文件 ): 在 UNIX 或 Linux 中 所 有 的 IO 设备 都 被 看 成 特殊 文 


件 , 通 过 链接 方式 , 它 与 设备 驱动 程序 紧密 相连 。 

5) 按 存 取 的 物理 结构 分 类 

顺序 (连续 ) 文 件 : 文件 中 的 记录 ,顺序 地 存储 到 连续 的 物理 块 中 ,顺序 文件 中 所 记录 的 
次 序 ,与 它们 存储 在 物理 介质 上 存放 的 顺序 是 一 致 的 。 例 如 ,存放 在 磁带 上 的 文件 。 

链接 文件 : 文件 中 的 记录 可 存放 在 并 不 相 邻 的 各 个 物理 块 中 ,通过 物理 块 中 的 链接 指 
针 组 成 一 个 链表 来 管理 ,形成 一 个 完整 的 文件 ,又 称 为 直接 存 取 文件 或 指针 串联 文件 。 

索引 文件 : 文件 的 记录 可 存储 在 并 不 相 邻 的 各 个 物理 块 中 ,记录 和 物理 块 之 间 通 过 索 
引 表 项 按 关 键 字 存 取 文件 ,通过 物理 块 中 的 索引 表 的 管理 ,形成 一 个 完整 的 文件 。 

6) 按 文件 的 逻辑 存储 结构 分 类 

有 结构 文件 : 由 若干 个 记录 所 构成 的 文件 ,又 称 为 记录 式 文件 。 根 据 记 录 的 长 度 特点 
又 可 分 为 定 长 记录 文件 和 可 变 长 记录 文件 。 例 如 ,目前 常用 的 数据 库 文件 大 多 是 定 长 记录 
文件 。 

无 结构 文件 : 这 是 直接 由 字符 序列 所 构成 的 文件 ,又 称 为 流 式 文件 。 一 般 来 说 ,操作 系 
统 就 是 这 种 文件 结构 ,可 以 把 流 式 文件 看 成 是 记录 文件 的 特例 , 即 其 中 每 个 记录 只 含有 一 个 
字符 。 

7) 按 文件 的 数据 形式 分 类 

源 文件 : 是 指 源 程序 和 数据 构成 的 文件 ,一 般 由 ASCII 码 `.EBCD 码 或 汉字 编码 组 成 。 

目标 文件 : 由 源 程序 经 过 相应 的 计算 机 编译 程序 编译 ,但 尚未 经 过 链接 程序 链接 时 的 
目标 代码 所 形成 的 文件 ,属于 二 进 制 文件 ,内 部 地 址 为 相对 地 址 。 

执行 文件 : 目标 文件 经 过 与 计算 机 系统 提供 的 库 函 数 及 相关 的 子 程序 链接 后 形成 的 文 
件 。 它 是 二 进 制 文件 ,可 在 操作 系统 的 支持 下 运行 。 

2. 文件 结构 

对 有 结构 文件 ,可 以 通过 自 底 向 上 的 方式 来 定义 文件 的 结构 。 自 底 向 上 依次 如 下 。 

1) 数据 项 

数据 项 是 文件 系统 中 最 低级 的 数据 组 织 形式 ,可 分 为 以 下 两 种 类 型 。 

基本 数据 项 : 用 于 描述 一 个 对 象 的 某 种 属性 的 一 个 值 , 如 姓名 、 日 期 或 证 件 号 等 ,是 数 
据 中 可 命名 的 最 小 逻辑 数据 单位 , 即 原子 数据 。 

组 合 数据 项 : 由 多 个 基本 数据 项 组 成 。 

2) 记录 

记录 是 一 组 相关 的 数据 项 的 集合 ,用 于 描述 一 个 对 象 在 某 方面 的 属性 ,如 一 个 考生 报名 
记录 包括 考生 姓名 、 出 生日 期 \ 报 考 学 校 代号 、 身 份 证 号 等 一 系列 域 。 

3) 文件 

文件 是 一 组 相似 记录 的 集合 ,被 用 户 和 应 用 程序 看 作 是 一 个 实体 ,并 可 以 通过 名 字 
访问 。 


7.1.3 文件 系统 的 功能 


操作 系统 中 与 管理 文件 有 关 的 软件 和 数据 称 为 文件 系统 。 文 件 系统 作为 一 个 统一 的 信 
息 管理 机 制 ,应 具有 下 述 功能 。 
(1) 统一 管理 文件 存储 空间 ( 即 外 存 ) ,实施 存储 空间 的 分 配 与 回收 。 即 在 用 户 创建 新 
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文件 时 为 其 分 配 空闲 区 ,而 在 用 户 删 除 或 修改 某 个 文件 时 ,回收 和 调整 存储 

(2) 确定 文件 信息 的 存放 位 置 及 存放 形式 。 

(3) 实现 文件 从 名 字 空 间 到 外 存 地 址 空间 的 映射 ,实现 文件 的 按 名 存 取 。 即 文件 有 一 
个 用 户 可 见 的 逻辑 结构 ,用 户 按照 文件 逻辑 结构 所 给 定 的 方式 进行 信息 的 存 取 和 加 工 ,并 且 
这 种 逻辑 结构 是 独立 于 物理 存储 设备 的 ,从 而 使 用 户 不 必 了 解 文件 存放 的 物理 结构 和 查找 
方法 等 与 存 取 介 质 有 关 的 部 分 ,只 需 给 定 一 个 代表 某 一 文件 的 文件 名 ,文件 系统 就 会 自动 地 
完成 对 与 给 定 文件 名 相对 应 文件 的 有 关 操 作 。 

(4) 有 效 实现 对 文件 的 各 种 控制 操作 (如 建立 撤销、 打开 、 关 闭 文件 等 ) 和 存 取 操作 (如 
读 , 写 ,修改 复制、 转 储 等 )。 

(5) 实现 文件 信息 的 共享 ,并 且 提 供 可 靠 的 文件 保密 和 保护 措施 。 


7.2 文件 组 织 和 存 取 


文件 组 织 指 文件 中 的 逻辑 结构 , 它 由 用 户 访问 记录 的 方式 确定 。 在 选择 文件 组 织 时 ,有 
5 项 重要 原则 : 访问 快速 .易于 修改 .节约 空间 、 维 护 简单 .可 靠 性 。 这 些 原则 的 相对 优先 级 
取决 于 将 要 使 用 这 些 文件 的 应 用 程序 。 这 些 原 则 可 能 是 矛盾 的 所 以 需要 根据 需要 进行 取 
合 。 常 见 的 文件 组 织 有 以 下 几 种 。 

1. 堆 

堆 是 最 简单 的 文件 组 织 形式 。 数 据 按 它们 到 达 的 顺序 被 采集 ,每 个 记录 由 一 串 数据 组 
成 。 堆 的 目的 仅仅 是 积累 大 量 的 数据 并 保存 数据 。 记 录 可 以 有 不 同 的 域 ,或 者 域 相似 但 是 
顺序 不 同 。 因 此 每 个 域 应 该 是 自 描述 的 ,包括 域名 和 值 。 每 个 域 的 长 度 由 划分 符 隐 式 地 指 
定 , 或 者 明确 地 包含 在 一 个 子 域 中 ,或 者 该 域 类 型 的 默认 长 度 。 由 于 堆 文件 没有 结构 ,因而 
对 记录 的 访问 是 通过 穷 举 查找 的 方式 ,也 就 是 说 如 果 想 要 找到 包括 某 一 指定 域 且 值 为 某 一 
特定 值 的 记录 , 则 需要 检查 堆 中 的 每 一 个 记录 ,直到 找到 想 要 的 记录 或 者 找到 完整 的 文件 为 
止 。 如 果 想 查找 包括 某 一 特定 域 ,或 者 包含 某 一 特定 值 的 域 的 所 有 记录 , 则 需 查 找 整个 文 
件 。 当 数据 在 处 理 前 采集 并 存储 时 ,或 者 当 数 据 难以 组 织 时 ,会 用 到 堆 文 件 。 当 保存 的 数据 
大 小 和 结构 不 同时 ,这 种 类 型 的 文件 空间 使 用 情况 良好 ,能 较 好 地 用 于 穷 举 查找 。 

2. 顺序 文件 

顺序 文件 是 最 常用 的 文件 组 织 形式 。 顺 序 文件 由 一 系列 记录 按照 某 种 顺序 排列 形成 。 
其 中 的 记录 通常 是 定 长 记录 ,因而 能 用 较 快 的 速度 查找 文件 中 的 记录 。 记 录 中 有 一 个 特殊 
的 域 , 通 常 是 每 条 记录 的 第 一 个 域 , 称 为 关键 域 。 关 键 域 唯一 地 标识 这 条 记录 ,因此 不 同 记 
录 的 关键 域 是 不 同 的 。 此 外 ,记录 按 关键 域 来 存储 : 文本 关键 域 按 字母 顺序 ,数字 关键 域 按 
照 数 字 顺 序 。 

顺序 文件 是 记录 按 其 在 文件 中 的 逻辑 顺序 依次 进入 存储 介质 而 建立 的 , 即 顺序 文件 中 
物理 记录 的 顺序 和 逻辑 记录 的 顺序 是 一 致 的 。 若 次 序 相继 的 两 个 物理 记录 在 存储 介质 上 的 
存储 位 置 是 相 邻 的 , 则 又 称 为 连续 文件 。 文 件 是 记录 的 集合 。 文 件 中 的 记录 可 以 是 任意 顺 
序 的 ,因此 , 它 可 以 按照 各 种 不 同 的 顺序 进行 排列 。 一 般 地 可 以 归纳 为 以 下 两 种 情况 : 第 一 
种 情况 是 串 结 构 , 各 记录 之 间 的 顺序 与 关键 字 无 关 。 通 常 的 办 法 是 由 时 间 来 决定 , 即 按 存 入 
时 间 的 先后 排列 ,最 先 存 入 的 记录 作为 第 一 个 记录 ,其 次 存 入 的 为 第 二 个 记录 …… 以 此 类 


区 


推 。 第 二 种 情况 是 顺序 结构 , 指 文件 中 的 所 有 记录 按 关键 字 ( 词 ) 排 列 。 可 以 按 关键 词 的 长 
短 从 小 到 大 排序 ,也 可 以 从 大 到 小 排序 ; 或 按 其 英文 字母 排序 。 

顺序 文件 通常 用 于 批 处 理应 用 中 ,并 且 如 果 这 类 应 用 涉及 对 所 有 记录 的 处 理 ( 如 关于 机 
长 或 工资 单 的 应 用 ), 则 顺序 文件 通常 是 最 佳 的。 顺序 文件 组 织 是 唯一 可 以 很 容易 地 存储 在 
磁盘 和 磁带 上 的 文件 组 织 。 

3。 论 引文 件 

由 索引 表 和 主 文件 两 部 分 构成 。 索 引 表 是 一 张 指示 逻辑 记录 和 物理 记录 之 间 对 应 关系 
的 表 。 索 引 表 中 的 每 项 称 作 索引 项 。 索 引 本 身 非 常 小 ,只 占 两 个 字段 : 顺序 文件 的 键 和 在 
磁盘 上 相应 记录 的 地 址 。 存 取 文 件 中 的 记录 需 按 以 下 步骤 。 

(1) 整个 索引 文件 都 载 入 到 内 存 中 (文件 很 小 ,只 占用 很 小 的 内 存 空间 ) 。 

(2) 搜索 项 目 , 用 高 效 的 算法 (如 折 半 查询 法 ) 查 找 目标 键 。 

(3) 检索 记录 的 地 址 。 

(4) 按照 地 址 ,检索 数据 记录 并 返回 给 用 户 。 

索引 文件 的 索引 项 是 按键 (或 逻辑 记录 号 ) 顺 序 排列 。 若 文件 本 身 也 是 按 关 键 字 顺 序 排 
列 , 则 称 为 索引 顺序 文件 。 否 则 , 称 为 索引 非 顺序 文件 。 

索引 文件 的 好 处 之 一 就 是 可 以 有 多 个 索引 ,每 个 索引 有 不 同 的 键 。 例 如 ,职员 的 文件 可 
以 按 社 会 保险 号 或 姓名 来 检索 。 这 种 索引 文件 被 称 为 倒 排 文件 。 

4.。 散 列 文件 

散 列 文件 类 似 于 散 列 表 , 但 与 散 列 表 不 同 的 是 ,对 于 文件 来 说 ,磁盘 上 的 文件 记录 通常 
是 成 组 存放 的 ,若干 个 记录 组 成 一 个 存储 单位 ,在 散 列 文件 中 ,这 个 存储 单位 叫 作 桶 
(Bucket)。 假 如 一 个 桶 能 存放 m 个 记录 , 则 当 桶 中 己 有 m 个 同义词 的 记录 时 ,存放 第 mm 十 1 
个 同义词 会 发 生 “ 溢 出 ”。 需 要 将 第 m 十 1 个 同义词 存放 到 另 一 个 桶 中 ,通常 称 此 桶 为 “溢出 
桶 ”"。 相 对 地 , 称 前 m 个 同义词 存放 的 桶 为 “ 基 桶 ”"。 处 理 溢出 虽 可 采用 散 列 表 中 处 理 冲 突 
的 各 种 方法 ,但 对 散 列 文件 而 言 ,主要 采用 拉链 法 。 

在 散 列 文件 中 进行 查找 时 ,首先 根据 给 定 值 求 出 散 列 桶 地 址 ,将 基 桶 的 记录 读 入 内 存 ， 
进行 顺序 查找 , 若 找到 关键 字 等 于 给 定 值 的 记录 , 则 检索 成 功 ; 否则 , 读 入 溢出 桶 的 记录 继 
续 进 行 查找 。 在 散 列 文件 中 删 去 一 个 记录 , 仅 需 对 被 删除 记录 标记 即 可 。 

散 列 文件 的 优点 是 : 文件 随机 存放 ,记录 不 需 进行 排序 ; 插入 、 删 除 方便 ; 存 取 速 度 快 ; 
不 需要 索引 区 ,节省 存储 空间 。 

缺点 是 : 不 能 进行 顺序 存 取 ,只 能 按 关键 字 随 机 存 取 , 且 询 问 方式 限于 简单 询问 ,并 且 
在 经 过 多 次 插入 、 删 除 后 ,也 可 能 造成 文件 结构 不 合理 ,需要 重新 组 织 文件 。 

5. B 树 

B 树 又 叫 平衡 多 路 查找 树 。B 树 中 的 每 个 结 点 根据 实际 情况 可 以 包含 大 量 的 关键 字 信 
息 和 分 支 ( 当 然 是 不 能 超过 磁盘 块 的 大 小 ,根据 磁盘 驱动 的 不 同 ,一 般 块 的 大 小 在 1 一 4kb 
左右 ); 这 样 树 的 深度 降低 了 ,就 意味 着 查找 一 个 元 素 只 要 很 少 结 点 从 外 存 磁盘 中 读 入 内 
存 , 就 能 很 快 访问 到 要 查找 的 数据 。 相 较 于 二 叉 树 的 优势 就 在 于 此 了 (降低 了 树 高 )。 

B 十 树 是 应 文件 系统 所 需 而 产生 的 一 种 B 树 的 变形 树 。 

一 棵 mm 阶 的 B 十 树 和 xm 阶 的 B 树 的 差异 在 于 : 

(1) 有 守 棵 子 树 的 结 点 中 含有 ?2 个 关键 字 ( 而 B 树 是 n 棵 子 树 有 nn 一 1 个 关键 字 ); 
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(2) 所 有 的 叶子 结 点 中 包含 全 部 关键 字 的 信息 ,及 指向 含有 这 些 关 键 字 记 录 的 指针 , 且 
叶子 结 点 本 身 依 关键 字 的 大 小 自 小 而 大 的 顺序 链接 (而 也 树 的 叶子 结 点 并 没有 包括 全 部 需 
要 查找 的 信息 ); 

(3) 所 有 的 非 终端 结 点 可 以 看 成 是 索引 部 分 , 结 点 中 仅 含 有 其 子 树 根 结 点 中 最 大 (或 最 
小 ) 关 键 字 ( 而 也 树 的 非 终结 点 也 包含 需要 查找 的 有 效 信息 ) 。 

B 十 树 的 优点 如 下 。 

1) B 十 树 的 磁盘 读 写 代 价 更 低 

B 十 树 的 内 部 结 点 并 没有 指向 关键 字 具 体 信息 的 指针 ,因此 其 内 部 结 点 相对 B 树 更 小 。 
如 果 把 所 有 同一 内 部 结 点 的 关键 字 存放 在 同一 盘 块 中 ,那么 盘 块 所 能 容纳 的 关键 字数 量 也 
越 多 ,一 次 性 读 入 内 存 中 的 需要 查找 的 关键 字 也 就 越 多 ,相对 来 说 ,I/O 读 写 次 数 也 就 降 
低 了 。 

2) B 十 树 的 查询 效率 更 加 稳定 

由 于 非 终结 点 并 不 是 最 终 指向 文件 内 容 的 结 点 ,而 只 是 叶子 结 点 中 关键 字 的 索引 ,所 以 
任何 关键 字 的 查找 必须 走 一 条 从 根 结 点 到 叶子 结 点 的 路 。 所 有 关键 字 查 询 的 路 径 长 度 相 
同 ,导致 每 一 个 数据 的 查询 效率 相当 。 

6. 记录 的 成 组 与 分 解 

磁盘 块 的 大 小 是 磁盘 的 固有 属性 ,在 磁盘 格式 化 的 时 候 就 已 经 确定 ,以 后 就 不 会 改变 
了 。 而 逻辑 记录 的 大 小 是 由 用 户 的 文件 性 质 决定 的 。 二 者 之 间 没 有 明确 的 对 应 关系 。 如 果 
逻辑 记录 比 物理 块 小 得 多 时 ,可 以 把 多 个 风 辑 记录 存放 在 一 个 块 中 ,这 就 是 记录 的 成 组 。 当 
用 户 使 用 时 再 从 读 取 的 一 块 信息 中 分 离 出 成 组 在 一 起 的 记录 ,从 中 找到 所 需 记录 ,这 就 是 记 
录 的 分 解 。 使 用 记录 的 成 组 与 分 解 技术 可 以 提高 存储 空间 的 利用 率 。 缺 点 是 由 于 存储 和 读 
取 的 过 程 中 需要 进行 处 理 , 所 以 对 于 运行 效率 会 有 一 定 程度 的 降低 。 


7.3 目录 管理 


一 个 计算 机 系统 中 有 成 千 上 万 个 文件 ,为 了 便于 对 文件 进行 存 取 和 管理 ,计算 机 系统 建 
立 了 文件 的 索引 , 即 文件 名 和 文件 物理 位 置 之 间 的 映射 关系 ,这 种 文件 的 索引 称 为 文件 
目录 。 


7.3.1 内 容 结构 


文件 目录 为 每 个 文件 设立 一 个 表 目 。 文 件 目录 表 目 至 少 要 包含 文件 名 ,文件 内 部 标识 、 
文件 的 类 型 ,文件 存储 地 址 文件 的 长 度 . 访 问 权限 、 建 立时 间 和 访问 时 间 等 内 容 。 

文件 目录 (或 称 为 文件 夹 ) 是 由 文件 目录 项 组 成 的 。 文 件 目录 分 为 一 级 目录 、 二 级 目录 
和 多 级 目录 。 多 级 目录 结构 也 称 为 树 状 结构 ,在 多 级 目录 结构 中 ,每 一 个 磁盘 有 一 个 根 目 
录 , 在 根 目录 中 可 以 包含 若干 子 目 录 和 文件 ,在 子 目录 中 不 但 可 以 包含 文件 ,还 可 以 包含 下 
一 级 子 目 录 , 这 样 类 推 下 去 就 构成 了 多 级 目录 结构 。 采 用 多 级 目录 结构 的 优点 是 用 户 可 以 
将 不 同类 型 和 不 同 功能 的 文件 分 类 储存 , 既 方 便 文件 管理 和 查找 ,还 允许 不 同文 件 目录 中 的 
文件 具有 相同 的 文件 名 ,解决 了 一 级 目录 结构 中 的 重 名 问题 。 


7.3.2 命名 


在 多 用 户 系 统 中 ,每 个 用 户 都 会 创建 自己 的 文件 。 为 了 实现 文件 的 按 名 访问 ,要 求 系统 
中 的 文件 具有 唯一 的 名 字 。 但 是 对 于 用 户 来 讲 ,为 文件 提供 唯一 的 名 字 是 非常 难 实现 的 。 
在 共享 系统 中 ,这 尤其 难以 实现 。 

使 用 多 级 目录 可 以 比较 好 地 解决 这 个 问题 ,在 不 同 目录 里 可 以 有 相同 名 字 的 文件 存在 。 
多 级 目录 具有 树 状 结构 ,由 于 系统 中 可 能 有 同名 文件 存在 ,所 以 在 访问 文件 时 除了 指定 文件 
名 字 之 外 还 需要 指定 文件 的 目录 、 目 录 的 目录 …… 将 从 根 目 录 开 始 的 目录 序列 加 上 最 后 访 
问 的 文件 名 放 到 一 起 叫 作 文件 的 路 径 名 。 在 使 用 路 径 名 的 过 程 中 需要 对 路 径 和 子路 径 进 行 
区 分 。 这 就 需要 使 用 特殊 符号 进行 分 隔 。 不 同系 统 中 使 用 的 符号 也 不 完全 相同 , 比 刀 
Windows 系统 使 用 “\”,Linux 和 UNIX 系统 使 用 “/”。 

使 用 路 径 名 解决 了 同名 文件 的 使 用 问题 。 但 是 路 径 名 的 使 用 ,也 加 重 了 用 户 的 负担 , 因 
为 每 次 使 用 文件 都 需要 用 户 输入 完整 的 路 径 名 。 为 了 简化 用 户 使 用 文件 ,操作 系统 使 用 了 
当前 目录 的 概念 。 当 用 户 访问 文件 时 ,通常 按照 相对 于 工作 目录 的 方式 被 访问 , 当 交 互 用 户 
登录 进来 时 ,或 者 当 创建 一 个 进程 时 ,默认 的 工作 目录 都 是 用 户 目录 。 执 行 过 程 中 用 户 可 以 
定义 不 同 的 当前 目录 。 


7.4 文件 共享 与 安全 


多 用 户 环境 下 ,有 些 文件 可 以 被 多 个 用 户 访问 。 不 同 的 用 户 对 文件 应 该 有 不 同 的 访问 
权限 ,访问 权限 应 该 由 操作 系统 进行 统一 的 管理 。 在 多 用 户 的 环境 下 还 会 产生 文件 的 安全 
性 问题 。 


7.4.1 访问 权限 


文件 共享 要 求 操 作 系统 提供 相应 的 工具 ,使 得 对 某 个 特定 的 文件 的 访问 方式 可 以 被 控 
制 。 典 型 情况 下 ,比如 UNIX 系统 中 ,访问 权限 可 以 被 授予 用 户 或 用 户 组 。 常 见 的 访问 权 
限 有 以 下 几 种 。 

无 权限 : 用 户 不 知道 文件 的 存在 ,更 无 法 访问 文件 。 为 了 实现 这 个 访问 权限 ,应 该 禁止 
用 户 访问 文件 所 在 目录 。 

知道 , 用 户 可 以 确定 文件 是 否 存在 以 及 文件 的 所 有 者 ,但 是 不 能 使 用 文件 。 为 了 获得 
更 高 权限 ,用 户 可 以 向 文件 的 所 有 者 提出 申请 。 

执行 : 用 户 可 以 加 载 并 执行 一 个 程序 ,但 是 不 能 复制 它 。 私 有 程序 通常 具有 这 种 访问 
限制 。 

读 : 用 户 能 够 以 任何 的 目的 读 文件 ,包括 复制 和 执行 。 有 些 系统 可 以 区 分 浏览 和 复制 ， 
对 于 前 一 种 情况 ,文件 的 内 容 可 以 呈现 给 用 户 , 但 用 户 没有 办 法 进行 复制 。 

追加 : 用 户 可 以 给 文件 添加 数据 ,通常 只 能 在 末尾 追加 ,但 不 能 修改 或 删除 文件 的 任何 
内 容 。 当 在 许多 资源 中 收集 数据 时 ,这 种 权限 非常 有 用 。 

更 新 : 用 户 可 以 修改 ,删除 和 增加 文件 中 的 数据 。 这 通常 包括 最 初 写 文件 .完全 重 写 文 
件 或 部 分 重 写 文件 ,删除 所 有 或 部 分 数据 。 
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改变 保护 : 用 户 可 以 改变 授予 其 他 用 户 的 访问 权限 。 一 般 情况 下 ,只 有 文件 的 所 有 者 
才 具 有 这 项 权利 。 
删除 : 用 户 可 以 从 系统 中 删除 该 文件 。 
上 述 权 限 由 上 至 下 构成 了 一 个 层次 ,下 面 的 每 一 项 都 隐 含 它 上 面 的 所 有 权限 ,因此 当 向 
户 授予 权限 时 ,只 要 授予 其 最 大 的 权限 即 可 。 
多 用 户 系统 中 ,也 可 以 按照 访问 权限 对 用 户 进行 分 类 ,主要 分 为 如 下 三 类 。 
特定 用 户 : 由 用 户 ID 指定 的 单个 用 户 。 
用 户 组 : 不 是 单个 用 户 定义 的 一 组 用 户 。 系 统 必 须 可 以 通过 某 种 方式 了 解 用 户 组 的 所 
有 成 员 。 
全 部 : 访问 该 系统 的 所 有 用 户 。 有 这 个 权限 的 文件 是 公共 文件 。 
一 般 会 有 一 个 用 户 被 指定 为 文件 的 所 有 者 ,通常 文件 的 所 有 者 就 是 文件 的 创建 者 。 文 
件 的 所 有 者 拥有 关于 文件 的 全 部 权限 ,并 且 可 以 给 其 他 用 户 授予 不 同 的 权限 。 


7.4.2 同时 访问 


如 果 多 个 用 户 具有 追加 或 者 更 新 一 个 文件 的 权限 ,操作 系统 或 者 文件 管理 器 必须 考 
虑 对 文件 加 锁 ,以 防止 多 个 用 户 同时 修改 造成 文件 的 不 一 致 。 这 跟 读 者 - 写 者 问题 中 多 个 
写 者 直接 互 斥 的 情况 是 一 样 的 。 只 是 在 文件 系统 中 可 以 设置 不 同 粒度 的 锁 , 以 避免 对 并 
发 性 造成 太 大 的 影响 。 需 要 注意 的 是 ,给 文件 加 锁 可 能 会 造成 死 锁 问 题 , 死 锁 将 在 第 8 
章 介绍 。 


7.4.3 文件 安全 


用 户 只 有 在 登录 系统 后 才 会 有 权限 访问 主机 和 主机 上 的 程序 。 对 于 存 有 敏感 数据 的 数 
据 库 系统 来 说 ,这 种 权限 管理 是 不 够 的 。 因 为 前 面 所 讲 的 访问 权限 是 针对 整个 文件 的 ,也 就 
是 用 户 登 录 以 后 ,系统 会 通过 访问 控制 程序 查找 用 户 相关 的 配置 文件 ,确定 用 户 的 访问 权 
限 , 一 旦 用 户 可 以 访问 一 个 文件 ,就 意味 着 用 户 可 以 访问 文件 的 全 部 内 容 。 但 是 数据 库 
系统 中 有 可 能 用 户 只 能 访问 其 中 的 一 部 分 信息 ,比如 所 有 登录 用 户 可 以 查看 公司 的 人 员 
列表 ,但 是 只 有 少数 有 权限 的 人 才能 查看 工资 信息 。 这 就 要 求 系统 能 够 细 化 信息 访问 控 
制 的 粒度 。 不 是 针对 整个 文件 设置 访问 权限 ,而 是 针对 文件 中 的 不 同 部 分 设置 权限 , 比 
如 可 以 针对 数据 库 中 的 特定 记录 或 者 一 部 分 记录 设置 权限 。 使 用 细 粒 度 权限 的 安全 控 
制 以 后 ,用 户 对 数据 的 访问 就 不 止 取决 于 访问 用 户 的 ID, 也 取决 于 用 户 要 访问 的 是 哪 一 
部 分 数据 。 

在 文件 或 数据 库 系 统 中 最 常 使 用 的 访问 控制 模型 叫 作 访问 矩阵 。 在 访问 矩阵 中 有 以 下 
三 种 基本 元 素 。 
主体 : 有 能 力 访 问 对 象 的 实体 。 一 般 来 说 ,实体 的 概念 等 同 于 进程 的 概念 ,任何 一 个 用 
户 或 应 用 程序 通过 代表 它们 自己 的 进程 来 获得 访问 权限 。 

对 象 : 可 以 被 访问 和 控制 的 任何 实体 ,比如 文件 .文件 局 部 数据 ,程序 .内存 块 ,以 及 软 
件 中 的 对 象 。 

访问 权限 : 主体 访问 对 象 的 方式 ,比如 读 、 写 .执行 以 及 使 用 软件 对 象 的 功能 。 

访问 矩阵 的 每 一 行 代表 一 个 主体 ,每 一 列 代表 一 个 对 象 ,矩阵 中 的 数据 存储 的 是 访问 权 


限 。 一 个 可 能 的 访问 矩阵 如 表 7-1 所 示 。 
表 7-1 访问 矩阵 


交 儿 工 交 和 天 & 文件 3 文件 4 账户 5 账户 6 
Own Own 信用 查询 
用 户 A R 
Ww 
用 户 B R Own Ww R 债务 查询 信用 查询 
R Own 
用 户 C Ww R R 债务 查询 
Ww 


从 表 中 可 以 看 出 , 表 中 并 不 是 所 有 位 置 都 是 填 满 的 ,实际 上 因为 系统 可 能 的 主体 和 对 象 
都 很 多 ,所 以 矩阵 通常 都 是 稀疏 矩阵 。 为 了 节省 空间 可 以 考虑 对 和 矩阵 进行 划分 。 和 矩阵 可 以 
按 列 划分 ,划分 后 就 生成 了 访问 控制 列表 。 对 于 每 个 对 象 ,一 个 访问 控制 列表 列 出 了 用 户 以 
及 他 们 的 访问 权限 。 访 问 控制 列表 包含 一 个 默认 或 公共 的 单元 。 这 允许 没有 明确 指出 有 哪 
些 权 限 的 用 户 具 有 默认 的 权限 。 

依照 行 划分 就 会 生成 权能 入 场 券 。 一 个 入 场 券 指 定 了 用 户 被 授权 的 对 象 和 操作 。 每 个 
用 户 有 许多 入 场 券 ,同时 可 以 授权 给 别人 。 因 为 系统 的 入 场 券 可 能 会 消失 ,这 就 意味 着 会 有 
比 访问 权限 更 大 的 安全 问题 ,尤其 是 用 户 的 入 场 券 可 能 是 伪造 的 。 为 了 解决 这 些 问 题 ,让 操 
作 系统 蔡 用 户 控制 入 场 券 可 能 是 一 种 很 好 的 方法 。 这 些 入 场 券 应 该 放 在 用 户 不 能 访问 的 
区 域 。 


7.5 辅 存 空间 管理 


在 辅 存 空间 中 ,一 个 文件 是 由 许多 块 组 成 的 。 操 作 系 统 或 文件 管理 系统 负责 给 文件 分 
配 块 。 由 此 带 来 两 个 问题 ,首先 辅 存 中 的 空间 必须 分 配给 文件 ; 其 次 必须 知道 哪些 空间 可 
以 用 来 分 配 。 下 面 分 别 讨论 这 两 个 问题 。 


7.5.1 文件 分 配 


辅 存 指 的 是 除了 CPU 缓存 和 内 存 以 外 的 存储 器 。 硬盘、 光盘 、U 盘 都 可 以 被 称 为 辅 
存 。 不 同 辅 存 都 是 以 块 为 单位 来 分 配 空间 的 。 所 以 下 面 统一 以 块 来 讨论 。 

外 存 分 配方 式 主 要 有 这 几 种 : 连续 分 配 , 链 式 分 配 , 索 引 分 配 。 

1. 连续 分 配 

连续 分 配方 式 是 把 一 个 在 逻辑 上 连续 的 文件 信息 顺序 地 存放 在 各 邻接 的 物理 块 中 ,如 
图 7-1 所 示 。 

优点 : 速度 快 ,节省 空间 。 

缺点 : 长 度 变化 困难 ,不 利于 文件 动态 扩充 ,存在 外 部 碎片 。 

说 明 : 该 方式 可 把 逻辑 文件 中 的 记录 ,顺序 地 存储 到 邻接 的 各 物理 块 中 ,这 样 所 形成 的 
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图 7-1 连续 分 配方 式 


文件 结构 称 为 顺序 文件 结构 ,此 时 的 物理 文件 称 为 顺序 文件 。 这 种 分 配方 式 保证 了 逻辑 文 
件 中 的 记录 顺序 与 存储 器 中 文件 占用 盘 块 的 顺序 的 一 致 性 。 

随 着 文件 的 建立 与 删除 不 断 进行 ,将 产生 很 多 外 碎片 ,利用 紧凑 方法 也 可 消除 碎片 。 

适用 于 : 变化 不 大 的 顺序 访问 的 文件 。 

2. 链接 分 配 

链接 分 配 是 一 种 离散 分 配方 式 ,可 将 文件 装 到 多 个 离散 的 盘 块 中 ,可 通过 在 每 个 盘 
块 上 的 链接 指针 ,将 同属 于 一 个 文件 的 多 个 离散 盘 块 链接 成 一 个 链表 。 这 样 形成 的 物理 
文件 就 是 链接 文件 。 链 接 方 式 分 为 : 隐 式 链接 , 显 式 链接 。 磁盘 空间 的 链接 式 分 配 如 
图 7-2 所 示 。 


directory 
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图 7-2 链接 分 配方 式 


1) 隐 式 链接 

在 隐 式 链接 中 一 个 串联 文件 结构 是 按 顺序 由 串联 的 块 组 成 的 , 即 文件 的 信息 存储 于 若 
干 块 中 。 每 个 物理 块 的 最 末 一 个 字 ( 或 第 一 个 字 ) 作 为 链接 字 , 它 指出 后 继 块 的 物理 地 址 。 
链 首 指针 存放 在 该 文件 目录 中 。 文 件 的 结尾 块 的 指针 为 * 人”。 这 种 文件 结构 不 要 求 连续 存 
放 。 对 于 记录 式 文件 一 块 中 可 包含 一 个 逻辑 记录 或 多 个 逻辑 记录 ,也 可 以 若干 物理 块 包含 
一 个 逻辑 记录 。 

无 外 部 碎片 ,没有 磁盘 空间 浪费 。 

无 须 事先 知道 文件 大 小 。 文 件 动 态 增长 时 ,可 动态 分 配 空闲 块 。 对 文件 的 增 、 删 、 改 十 
分 方便 。 

缺点 : 

不 能 支持 高 效 随机 /直接 访问 , 仅 对 顺序 存 取 有 效 , 顺 序 存 取 效率 高 ,随机 存 取 效 率 太 
低 , 如 果 访 问 文件 最 后 的 内 容 , 实 际 上 是 要 访问 整个 文件 。 

需 为 指针 分 配 空间 。 解 决 方法 是 将 几 个 盘 块 组 成 一 个 簇 ,降低 指针 在 分 配 空间 中 的 
比例 。 

可 靠 性 较 低 (指针 丢失 /损害 ) ,因此 引入 显 式 链接 解决 此 问题 。 

2) 显 式 链接 

显 式 链接 是 指 把 用 于 链接 文件 各 物理 块 的 指针 , 显 式 地 存放 在 内 存 的 一 张 链接 表 中 。 
该 表 在 整个 磁盘 中 就 一 张 。 在 DOS 和 Windows 中 这 个 表 就 是 文件 分 配 表 (FAT)。 文 件 分 
配 表 具 有 以 下 特征 : 用 于 链接 文件 各 物理 块 的 链接 指针 , 显 式 地 存放 在 内 存 的 一 张 链接 表 
中 ; 该 表 在 整个 磁盘 中 仅 设置 一 张 ; 表 序 号 为 整个 磁盘 的 物理 块 号 (0 一 n 一 1),n 是 盘 块 总 
数 ; 表 项 存 入 链接 指针 , 即 下 一 个 块 号 。 文 件 的 首 块 号 存 入 相应 文件 的 FCB 中 ; 查找 在 内 
存 的 FAT 中 , 故 提高 了 检索 速度 ,同时 又 减少 磁盘 的 访问 次 数 。 事 实 上 ,在 打开 某 个 文件 
时 ,只 需 把 该 文件 占用 的 盘 块 的 编号 调 入 内 存 即 可 ,完全 没有 必要 调 入 整个 FAT。 为 此 应 
将 每 个 文件 所 对 应 的 盘 块 号 集中 在 一 起 。 

3. 索引 分 配 

索引 分 配 为 每 个 文件 分 配 一 个 索引 块 ( 表 ) ,再 把 分 配给 该 文件 的 所 有 盘 块 号 都 记录 在 
该 索引 块 中 ,因而 该 索引 块 就 是 一 个 含有 许多 盘 块 号 的 数组 。 在 建立 一 个 文件 时 , 便 须 在 为 
之 建立 的 文件 目录 项 中 填 上 指向 该 索引 块 ( 表 ) 的 指针 。 索 引 分 配 为 每 个 文件 建立 一 个 索引 
表 , 表 中 每 个 表 目 包括 : 逻辑 块 号 ,与 该 逻辑 块 号 对 应 的 物理 块 号 。 索 引 表 可 以 放 在 文件 目 
录 中 文件 的 开头 等 。 索 引 结构 分 配方 式 如 图 7-3 所 示 。 由 于 索引 表 的 大 小 受 块 大 小 限制 ， 
索引 分 配方 法 能 够 创建 的 文件 大 小 也 就 受到 了 限制 ,为 了 解决 这 个 问题 。 可 将 索引 方式 进 
行 扩 展 , 这 就 产生 了 多 级 索引 和 混合 索引 的 概念 。 

1) 单 级 索引 分 配 
单 级 索引 分 配 是 只 有 一 级 索引 的 分 配方 式 , 如 图 7-4 所 示 。 

2) 多 极 索引 分 配 

当 操作 系统 为 一 个 大 文件 分 配 磁盘 空间 时 ,如 果 所 分 配 出 去 的 盘 块 的 盘 块 号 已 经 装 满 
一 个 索引 块 时 ,操作 系统 便 为 该 文件 分 配 另 一 个 索引 块 ,用 于 将 以 后 继续 为 之 分 配 的 盘 块 号 
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图 7-3 索引 分 配方 式 
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图 7-4 一 级 索引 分 配 


记录 于 其 中 。 以 此 类 推 , 再 通过 链 指针 将 各 索引 按 序 链 接 起 来 。 显 然 当 文 件 太 大 时 索引 块 
太 多 ,此 方法 低 效 。 此 时 应 为 索引 块 再 建立 一 级 索引 ,这 样 便 形 成 了 两 级 索引 分 配方 式 。 如 
果 文 件 非常 大 时 ,还 可 用 三 级 四 级 索引 方式 ,如 图 7-5 所 示 。 

3) 混合 索引 分 配方 式 

混合 索引 分 配方 式 : 是 指 将 多 种 索引 分 配方 式 结合 而 形成 的 一 种 分 配方 式 ,如 图 7-6 
所 示 。 混 合 索引 既 适 应 于 顺序 访问 ,也 适应 于 随机 访问 ,但 是 也 具有 索引 表 的 空间 开销 和 文 
件 索引 的 时 间 开销 。 


7.5.2 空闲 空间 管理 
文件 分 配 的 存储 空间 是 辅 存 中 的 空闲 空间 , 辅 存 中 的 空闲 空间 应 该 由 操作 系统 统一 进 
行 管理 ,常用 的 方法 主要 有 空闲 表 法 、 空 闲 链表 法 、 位 示 图 法 和 成 组 链接 法 。 
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图 7-6 混合 索引 方式 


1. 空闲 表 法 (空闲 文件 目录 ) 
操作 系统 为 磁盘 外 存 上 所 有 空闲 区 建立 一 张 空闲 表 , 每 个 表 项 对 应 一 个 空闲 
中 包含 序号 、 空 闲 区 的 第 一 块 号 ` 空 闲 块 的 块 数 等 信息 。 


区 ,空闲 表 


空闲 表 法 适用 于 连续 文件 结构 ,其 分 配 的 方式 是 ,在 系统 为 某 个 文件 分 配 空 闲 块 时 , 首 
先 扫 描 空 闲 表 项 ,如 找到 合适 的 空闲 区 项 , 则 分 配给 申请 者 ,并 把 该 项 从 空闲 表 中 去 掉 。 如 
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果 一 个 空闲 区 项 不 能 满足 申请 者 的 要 求 , 则 把 空闲 表 中 的 另 一 项 满足 要 求 的 分 区 分 配给 申 
请 者 (连续 文件 结构 除外 ) 。 如 果 一 个 空闲 表 项 所 含 块 数 超过 申请 者 要 求 , 则 为 申请 者 分 配 
了 所 要 的 物理 块 后 ,再 修改 该 表 项 。 

2. 空闲 链表 法 (自由 链表 法 ) 

将 所 有 空闲 盘 区 拉 成 一 条 空闲 链 , 根 据 空闲 链 所 有 的 基本 元 素 不 同 ,可 以 把 链表 分 成 两 
种 形式 ; 空闲 盘 块 链 和 空闲 盘 区 链 。 

空闲 盘 块 链 是 将 磁盘 上 的 所 有 空闲 分 区 ,以 盘 块 为 单位 拉 成 一 条 链 , 当 用 户 创建 文件 请 
求 分 配 存储 空间 时 ,系统 从 链 首 开始 ,依次 摘 下 适当 数目 的 空闲 盘 块 分 配给 用 户 ; 当 用 户 删 
除 文件 时 ,系统 将 回收 的 盘 块 依次 加 入 到 空闲 盘 块 链 的 末尾 。 

空闲 盘 区 链 是 将 磁盘 上 的 所 有 空闲 盘 区 (每 个 空闲 盘 区 可 包含 若干 个 盘 块 ) 拉 成 一 条 
链 。 在 管理 的 线性 表 中 ,每 一 个 表 项 对 应 一 个 空闲 区 ,增加 一 项 存放 指向 空闲 块 的 指针 ,将 
磁盘 上 的 所 有 空闲 区 (可 包含 若干 个 空闲 块 ) 拉 成 一 条 链 。 每 个 空闲 区 上 除 含 有 用 于 指示 下 
一 个 空闲 区 的 指针 外 ,还 有 本 盘 区 大 小 ( 盘 块 数 ) 的 信息 。 

3. 位 示 图 法 

本 方法 利用 二 进 制 的 一 位 来 表示 磁盘 中 一 个 盘 块 的 使 用 情况 ,磁盘 上 所 有 的 盘 块 都 有 
与 之 对 应 的 一 个 二 进 制 位 。 当 其 值 为 0 时 ,表示 对 应 的 盘 块 空闲 , 当 其 值 为 1 时 ,表示 对 应 
的 盘 块 已 经 分 配 。 

4. 成 组 链接 法 

空闲 表 法 和 空闲 链表 由 于 空闲 表 太 长 而 不 适合 大 型 文件 系统 的 使 用 。 成 组 链接 法 是 两 
种 方法 相 结合 的 一 种 管理 方法 ,兼备 了 两 种 方法 的 优点 而 克服 了 两 种 方法 的 缺点 。 其 大 致 
的 思想 是 : 把 空闲 的 ?个 空闲 扇 区 的 地 址 保存 在 第 一 个 空闲 扇 区 内 ,其 后 一 个 空闲 扇 区 内 
则 保存 另 一 顺序 空闲 扇 区 的 地 址 ,以 此 类 推 , 直 至 所 有 空闲 扇 区 都 予以 链接 。 


7.6 文件 的 使 用 


用 户 或 者 用 户 程 序 使 用 文件 时 都 需要 通过 路 径 名 找到 文件 的 属性 和 文件 的 位 置 ,然后 
对 文件 执行 读 写 操作 。 由 于 辅 存 的 速度 比 处 理 机 的 速度 低 很 多 ,所 以 多 次 辅 存 访 问 会 显著 
地 降低 执行 速度 。 当 多 次 访问 文件 时 ,效率 降低 的 更 明显 。 为 了 提高 效率 ,操作 系统 一 般 
都 提供 了 打开 文件 的 功能 。 打 开 文 件 会 将 文件 在 辅 存 的 位 置 以 及 文件 的 属性 等 信息 存 
储 在 操作 系统 的 内 存 表 中 ,并 将 内 存 表 中 的 编号 返回 给 用 户 。 以 后 当 用 户 再 需要 访问 该 
文件 时 可 以 直接 使 用 文件 在 内 存 表 中 的 编号 来 访问 ,从 而 提高 了 效率 。 内 存 表 在 操作 系 
统 也 占用 了 内 存 资 源 ,所 以 操作 系统 提供 了 跟 打开 文件 相对 应 的 关闭 文件 操作 来 释放 内 
存 资源 。 


小 结 
现代 操作 系统 都 需要 使 用 大 量 的 程序 和 数据 。 程 序 和 数据 要 求 大 容量 , 非 易 失 的 存 
储 器 。 
由 于 文件 系统 涉及 多 种 不 同 的 存储 器 和 不 同 的 文件 类 型 ,所 以 文件 系统 使 用 了 分 层 的 


结构 。 可 以 从 逻辑 结构 和 物理 结构 两 方面 来 理解 文件 。 


全 性 都 是 操作 系统 中 的 重要 内 容 。 基 于 访问 权限 和 力 


系统 中 存储 的 大 量 文件 需要 使 用 目录 来 进行 管理 。 多 用 户 操作 系统 中 文件 的 共享 和 安 


问题 。 
辅 存 中 存储 了 大 量 文件 ,文件 的 空间 分 配 和 空闲 空间 的 分 配 都 是 操作 系统 中 的 课题 。 


使 用 


内 存 中 的 打开 文件 表 可 以 提高 文件 的 访问 速度 。 


0 锁 的 方法 可 以 较 好 地 解决 共享 和 安全 


文件 管理 
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第 8 章 死 锁 


在 计算 机 系统 中 有 很 多 独占 性 的 资源 ,如 打印 机 、 磁 带 机 ,一 个 文件 的 i 结 点 等 。 在 多 
道 程序 环境 中 ,若干 进程 要 共用 这 类 资源 ,而 且 一 个 进程 所 需 的 资源 往往 不 止 一 个 。 这 样 就 
会 出 现 若干 进程 竞争 有 限 的 资源 ,又 加 上 推进 顺序 不 当 , 从 而 导致 进程 无 限 循 环 等 待 的 局 
面 , 即 死 锁 状 态 。 

死 锁 是 多 道 程序 并 发 执行 带 来 的 又 一 严重 问题 , 它 是 操作 系统 乃至 并 发 程序 设计 中 最 
难处 理 的 问题 之 一 。 处 理 不 好 会 导致 整个 系统 运行 效率 下 降 , 甚 至 不 能 正常 运行 。 

本 章 主 要 讲解 死 锁 的 原理 、 处 理 死 锁 的 方法 ,学 完 本 章 , 读 者 会 发 现 死 锁 普遍 存在 ,对 于 
死 锁 不 存在 完美 的 彻底 的 解决 方案 。 所 以 本 章 将 会 从 不 同 的 角度 给 出 处 理 死 锁 的 策略 , 即 
死 锁 的 检测 和 恢复 、 避 免 和 预防 。 最 后 介绍 了 和 死 锁 相 近 的 概念 : 活 锁 和 饥饿 。 


8.0 问题 导入 


假设 系统 中 有 P1,P2 两 个 进程 并 发 执行 ,Pl 和 P2 在 执行 中 同时 需要 资源 Rl 和 R2， 
R1 和 R2 都 是 一 次 只 能 给 一 个 进程 使 用 的 临界 资源 ,如 下 所 示 。 


P1() P2() 

{ { 

申请 R1; 申请 R2; 

申请 R2; 申请 R1; 

使 用 R1 和 R2; 使 用 Rl 和 R2; 
释放 R1 和 R2; 释放 Rl 和 R2; 


上 } 


如 果 系 统 对 资源 的 分 配 采 用 的 是 动态 分 配 且 先 申请 者 先 获得 ,就 有 可 能 出 现 以 下 的 运 
行 过 程 : 

P1 申请 并 获得 R1。 

P2 申请 并 获得 R2。 

P1 申请 R2, 因 得 不 到 而 阻塞 。 

P2 申请 R1, 因 得 不 到 而 阻塞 。 

此 时 ,系统 中 仅 有 的 两 个 进程 都 处 于 阻塞 状态 ,而 且 它们 都 在 等 待 对 方 释放 它们 所 需要 
的 资源 。 显 然 ,这 种 等 待 是 没有 止境 的 。 这 就 是 死 锁 。 

死 锁 是 指 系统 中 一 部 分 并 发 进程 彼此 相互 等 待 对 方 所 拥有 的 资源 ,而 且 这 些 并 发 进程 


在 没有 得 到 对 方 占有 的 资源 之 前 又 不 会 释放 自己 拥有 的 资源 ,从 而 导致 参与 的 进程 都 不 能 
继续 向 前 推进 的 一 种 系统 状态 。 
到 底 是 什么 导致 了 死 锁 ? 如 何 解决 死 锁 ? 本 章 的 后 面 几 节 将 加 以 详细 的 介绍 。 


8.1 死 锁 原理 


死 锁 是 多 个 进程 的 永久 性 阻塞 状态 ,产生 的 原因 主要 有 两 个 : 进程 间 竞争 资源 和 进程 
推进 顺序 非法 。 所 有 的 死 锁 都 涉及 两 个 或 更 多 的 进程 由 于 竞争 资源 引起 的 冲突 ,所 以 在 本 
节 中 将 从 资源 的 分 类 开始 介绍 ,最 后 介绍 死 锁 产生 的 必要 条 件 。 


8.1.1 资源 分 类 


大 部 分 死 锁 都 和 资源 有 关 , 资 源 是 指 系统 能 够 提供 给 用 户 进程 使 用 的 全 部 硬件 .软件 和 
数据 。 通 常 一 个 系统 拥有 一 定数 量 的 资源 ,分 布 在 若干 竞争 进程 之 间 。 这 些 资 源 可 分 成 多 
种 类 型 ,如 内 存 空 间 文件 .I/O 设备 等 。 每 种 类 型 有 一 定数 量 的 实例 ,如 系统 有 两 个 CPU， 
那么 资源 类 型 CPU 就 有 两 个 实例 。 当 某 一 资源 类 型 有 若干 个 实例 时 ,其 中 任何 一 个 都 可 
以 满足 对 资源 的 请 求 。 

系统 中 的 资源 类 型 通常 可 分 为 两 大 类 : 可 抢占 资源 和 不 可 抢占 资源 。 

1. 可 抢占 资源 

所 谓 “ 可 抢占 资源 ”, 是 指 可 以 从 拥有 它 的 进程 手中 抢夺 过 来 而 不 会 产生 副作用 的 那些 
资源 。 例 如 ,内 存储 器 就 是 一 种 可 抢占 资源 。 内 存 区 可 由 存储 器 管理 程序 把 一 个 进程 从 
个 存储 区 移 至 另 一 个 存储 区 , 即 剥 夺 了 该 进程 原来 占有 的 存储 区 ;， 甚至 可 将 一 个 进程 从 内 
存 调 到 外 存 上 。 可 见 , 内 存 属于 可 抢占 资源 。 

2. 不 可 抢占 资源 

所 谓 “ 不 可 抢占 资源 ”, 是 指 在 不 引起 相关 的 计算 失败 的 情况 下 ,无 法 把 它 从 占有 它 的 进 
程 处 抢占 过 来 。 例 如 ,一 个 进程 已 经 开始 刻 盘 , 突 然 将 CD 刻录 机 分 配给 另 一 个 进程 ,那么 
将 划 坏 CD 盘 , 所 以 在 任何 时 刻 CD 刻录 机 都 是 不 可 抢占 的 。 

一 般 来 说 , 死 锁 和 不 可 抢占 资源 有 关 , 有 关 可 抢占 资源 的 潜在 死 锁 通常 可 以 通过 在 进程 
之 间 重 新 分 配 资源 而 化 解 。 所 以 我 们 的 重点 是 考虑 不 可 抢占 资源 。 

在 正常 的 操作 模式 下 ,进程 按 如 下 顺序 使 用 资源 。 

(1) 申请 : 如 果 申 请 不 能 立即 被 允许 ,那么 申请 进程 必须 等 待 , 直 到 它 获得 该 资源 
为 止 。 

(2) 使 用 : 进程 对 资源 进行 操作 。 

(3) 释放 : 进程 释放 资源 。 

当 一 组 进程 中 的 每 个 进程 都 在 等 待 一 个 事件 ,而 这 一 事件 只 能 由 这 一 组 进程 的 另 一 进 
程 引起 ,那么 这 组 进程 就 处 于 死 锁 状 态 。 而 这 里 所 关心 的 主要 事件 就 是 资源 获取 与 释放 。 
例如 , 某 一 系统 有 一 台 打 印 机 和 一 台 DVD 驱动 器 。 假 如 进程 P; 占有 DVD 驱动 器 而 P; 占 
有 打印 机 ,如 果 P; 申请 打印 机 而 P; 申请 DVD 驱动 器 ,由 于 打印 机 和 DVD 驱动 器 是 不 可 
抢占 资源 ,那么 就 会 出 现 死 锁 。 
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8.1.2 资源 分 配 图 


死 锁 问题 可 用 称 为 系统 资源 分 配 图 的 有 向 图 进行 更 为 精确 的 描述 。 其 用 来 勾勒 系统 中 
各 个 进程 的 资源 分 配 情况 ,从 中 反映 哪个 进程 已 经 分 配 了 什么 资源 ,哪个 进程 由 于 等 候 什 么 
资源 而 处 于 阻塞 态 。 这 种 图 由 一 个 结 点 集合 V 和 一 个 边 集合 E 组 成 。 结 点 集合 V 分 为 两 
种 类 型 的 结 点 : P= 二 {Pi,Ps，,…,P,}( 系 统 活动 进程 的 集合 ) 和 R= 二 {Ri,R:,… ,Rm) (系统 所 
有 资源 类 型 的 集合 )。 

由 进程 P, 到 资源 类 型 Ri 的 有 向 边 记 为 ~ Pi, Ri >, 表 示 进 程 Pi 已 经 申请 资源 类 型 R; 
的 一 个 实例 ,并 正在 等 待 该 资源 ,由 资源 类 型 Ri 到 进程 P; 的 有 向 边 记 为 < Rj,P; >, 表 示 资 
源 类 型 Ri 的 一 个 实例 已 经 分 配给 进程 P:。 有 向 边 < Pi, Ri > 称 为 申请 边 , 有 向 边 < Rj,P; > 

通常 ,用 圆 形 表示 进程 P;, 用 和 矩形 表示 资源 类 型 R;。 由 于 资源 类 型 Ri 可 能 有 多 个 实 
例 , 所 以 在 矩形 中 用 圆 点 表示 实例 。 注 意 请 求 边 只 能 指向 矩形 Ri ,而 分 配 边 必须 指向 矩形 
内 的 某 个 圆 点 。 

例如 ,有 三 个 进程 Pi ,Ps,P 和 4 个 资源 Ri ,Ra,Rs ,Ri ,其 对 应 的 实例 个 数 分 别 是 1,2， 
1,3。 目 前 进程 的 状态 为 进程 Pi 占有 资源 类 型 Rs 的 一 个 实例 ,等待 资源 类 型 Ri 的 一 个 实 
例 ; 进程 P; 占有 资源 类 型 R 和 Rs* 各 一 个 实例 ,等 待 资源 类 型 R 的 一 个 实例 ; 进程 P; 占 
有 资源 类 型 Rs 的 一 个 实例 ,那么 集合 P= {Pi,Pz,Ps)},R={ Ri,R:,Rs,Ri} 和 E={<Pi，, 
Ri >,< P: ,Rs >,< Ri,P: >,< R:,P: >,< R: ,Pi >,< Rs ,P:, >} 构 成 的 资源 分 配 图 如 图 8-1 
所 示 。 


RI Rs 


图 8-1 资源 分 配 图 


根据 资源 分 配 图 的 定义 ,可 以 证 明 如 果 分 配 图 没有 环 ,那么 系统 就 没有 进程 死 锁 。 如 果 
分 配 图 有 环 ,那么 可 能 存在 死 锁 。 

如 果 资 源 分 配 图 中 出 现 了 环 , 且 处 于 此 环 中 的 每 个 资源 类 型 均 只 有 一 个 实例 , 则 有 环 就 
出 现 了 死 锁 。 此 时 , 环 是 系统 存在 死 锁 的 充 要 条 件 。 

如 果 资 源 分 配 图 中 出 现 了 环 , 但 是 处 于 此 环 中 的 资源 类 型 有 多 个 实例 ,那么 有 环 并 不 意 
味 着 已 经 出 现 了 死 锁 ,在 这 种 情况 下 , 环 就 是 死 锁 存 在 的 必要 条 件 而 不 是 充分 条 件 。 


例如 ,图 8-2 和 图 8-3 中 ,都 存在 环 Pi 一 Ri 一 P? 一 人 :一 Pi ,图 8-2 中 进程 P: 和 Ps 是 死 
锁 ,进程 Pi 等 待 资源 类 型 Ri ,而 它 又 被 进程 P。 占有 ,进程 Ps 等 待 进程 Pi 释放 的 资源 类 型 
R; ,构成 了 死 锁 ,然而 ,图 8-3 的 进程 却 没 有 死 锁 ,因为 Ps 可 以 请 求 到 资源 类 型 R; 的 实例 ， 
以 打破 环 。 


图 8-2 存在 死 锁 的 资源 分 配 图 图 8-3 存在 环 但 没有 死 锁 的 资源 分 配 图 


总 之 ,如 果 资 源 分 配 图 没有 环 , 那 么 系统 就 不 处 于 死 锁 状态 , 另 一 方面 ,如 果 有 环 ,那么 
系统 可 能 处 于 死 锁 状态 ,这 也 是 8. 1. 3 节 将 要 介绍 的 必要 条 件 之 一 。 


8.1.3 死 锁 的 必要 条 件 


如 果 在 系统 中 同时 满足 下 面 4 个 条 件 , 那 么 会 引起 死 锁 。 

(1) 互 斥 条 件 。 指 进程 对 所 分 配 到 的 资源 进行 排他 性 使 用 , 即 一 个 资源 每 次 只 能 被 一 
个 进程 使 用 。 

(2) 占有 且 等 待 条 件 。 一 个 进程 在 请 求 新 资源 而 阻塞 时 ,对 已 获得 资源 又 保持 不 放 。 
也 就 是 说 ,进程 并 不 是 一 次 性 得 到 所 需要 的 所 有 资源 ,而 是 在 占有 一 部 分 资源 的 情况 下 仍 允 
许 继续 申请 新 的 资源 。 

(3) 不 可 抢占 条 件 。 已 经 分 配 的 资源 不 能 从 相应 的 进程 中 被 强制 抢占 , 即 资源 只 能 在 
进程 完成 任务 后 自动 释放 。 

(4) 环 路 等 竺 条件。 在 发 生死 锁 时 ,必然 存在 一 个 进程 -资源 的 环 状 链 。 即 进程 集合 
{Pi ,Ps ,Pu} 中 的 P 正在 等 待 P: 占用 的 资源 ,Ps 正在 等 待 P 占用 的 资源 ,…,P, 正在 等 
待 Pi 占用 的 资源 。 也 就 是 说 ,在 多 个 进程 之 间 , 由 于 对 资源 的 占用 和 请 求 关系 而 形成 了 一 
个 循环 等 待 的 态势 。 

上 述 的 前 三 个 条 件 的 存在 起 到 决定 的 作用 ,第 4 个 条 件 实际 上 是 前 三 个 条 件 的 潜在 结 
果 , 它 取决 于 所 涉及 的 进程 请 求 和 释放 资源 的 顺序 。 

为 了 解决 死 锁 问题 ,可 使 用 下 面 几 种 对 策 。 

(1) 死 锁 的 预防 : 破坏 产生 死 锁 的 4 个 必要 条 件 中 的 一 个 或 多 个 ,使 系统 不 会 进入 死 锁 


(2) 死 锁 的 避免 : 在 资源 动态 分 配 过 程 中 使 用 某 种 算法 防止 系统 进入 不 安全 状态 ,从 


攻 o 避 


所 作 系 统 原 理 


而 避免 死 锁 的 发 生 。 

(3) 死 锁 的 检测 : 通过 系统 所 设置 的 检测 机 构 , 及 时 地 检测 出 死 锁 的 发 生 , 并 精确 地 确 
定 与 死 锁 有 关 的 进程 和 资源 ,然后 ,采取 适当 措施 ,从 系统 中 将 已 发 生 的 死 锁 清除 掉 。 

(4) 死 锁 的 解除 : 撤销 或 挂 起 一 些 进程 ,以 便 回收 一 些 资源 ,再 将 这 些 资 源 分 配给 已 处 
于 阻塞 态 的 进程 ,使 之 转 为 就 绪 态 ,以 继续 运行 。 

接 下 来 将 在 8. 2 一 8. 4 节 详 细 讨 论 每 种 处 理 方法 。 


8.2 死 锁 检 测 


所 谓 “ 死 锁 检 测 ”, 即 系统 允许 产生 死 锁 ,操作 系统 周期 性 地 在 进程 和 资源 之 间 检 测 是 否 
出 现 了 循环 等 待 的 情形 。 若 检测 到 死 锁 则 采取 相应 的 办 法 解除 死 锁 , 以 尽 可 能 小 的 代价 恢 
复 死 锁 进 程 的 运行 。 系 统 为 了 提供 检测 和 解除 死 锁 ,必须 做 到 保存 有 关 资 源 的 请 求 和 分 配 
信息 ,并 提供 一 种 算法 ,以 利用 这 些 信 息 来 检测 系统 是 否 进入 死 锁 状态 。 下 面 将 重点 介绍 死 
锁 检测 的 算法 ,以 及 如 何 进行 死 锁 的 修复 。 


8.2.1 死 锁 检 测算 法 


1. 死 锁定 理 

在 介绍 具体 的 死 锁 检测 算法 之 前 , 先 来 学 习 一 下 死 锁定 理 。 在 前 面 介绍 的 资源 分 配 图 
中 我 们 已 经 了 解 资 源 分 配 图 和 死 锁 的 关系 ,我 们 可 以 利用 下 述 步 又 运行 一 个 “ 死 锁 检 测 ” 程 
序 , 对 资源 分 配 图 进行 分 析 和 简化 ,以 此 方法 来 检测 系统 是 否 处 于 死 锁 状态 。 

(1) 如 果 能 在 资源 分 配 图 中 找 出 一 个 既 不 阻塞 又 非 独 立 的 进程 , 它 在 有 限 的 时 间 内 有 
可 能 获得 所 需 资源 类 中 的 资源 继续 执行 ,直到 运行 结束 ,再 释放 其 占有 的 全 部 资源 。 相 当 于 
消去 了 图 中 此 进程 的 所 有 请 求 边 和 分 配 边 ,使 之 成 为 孤立 结 点 ,如 将 如 图 8-4(a) 所 示 的 进程 
Pi 的 所 有 请 求 边 和 分 配 边 消去 ,使 得 进程 Pi 成 为 孤立 结 点 。 


Ri RI 击 
. 二 


图 8-4 资源 分 配 图 的 简化 


(2) 可 使 资源 分 配 图 中 另 一 个 进程 获得 前 面 进程 释放 的 资源 继续 执行 ,直到 完成 后 释 
放出 它 所 占用 的 所 有 资源 ,相当 于 又 消去 了 图 中 若干 请 求 边 和 分 配 边 。 如 将 如 图 8-4(b) 所 
示 的 进程 Ps 的 所 有 请 求 边 和 分 配 边 消去 ,使 得 进程 P。 成 为 孤立 结 点 。 

(3) 如 此 下 去 ,经 过 一 系列 简化 后 , 若 能 消去 图 中 所 有 边 , 使 所 有 进程 成 为 孤立 结 点 , 则 
该 图 是 可 完全 简化 的 ,如 图 8-4(c) 所 示 ; 否则 称 该 图 是 不 可 完全 简化 的 。 


系统 处 于 死 锁 的 充分 条 件 是 : 当 且 仅 当 此 状态 资源 分 配 图 是 不 可 完全 简化 的 ,ii 
分 条 件 称 为 死 锁 定理 。 
2. 检测 算法 


下 面 介绍 一 种 具体 的 死 锁 检 测 方法 ,该 算法 由 Shoshani 和 Coffman 提出 , 采 


结构 如 下 。 


目的 数据 


(1) 当前 可 分 配 的 空闲 资源 向 量 Available, 其 为 长 度 为 m 的 向 量 。m 是 系统 中 的 资源 


类 型 数 。Available[i] 表 示 系 统 中 现 有 的 i 类 资源 数量 。 


(2) 资源 分 配 和 矩阵 Allocation, 其 为 nXm 和 矩阵。Allocation[z, 门 表示 进程 z 已 占有 的 了 


类 资源 的 数量 。 

(3) 需求 矩阵 Request, 其 为 nXm 和 矩阵。Request[i, 站 表示 进程 i 还 需 申请 j 类 资源 的 
数量 。 

死 锁 检测 算法 如 下 。 


(1) 令 Work 和 Finish 分 别 表示 长 度 为 mx 和 nn 的 向 量 , 初 始 化 Work 二 Available, 对 于 
所 有 i==1,…,n, 如 果 Allocation [i, x ] 关 0, 则 Finish[ 门 =false, 否 则 Finish[ 让 一 true。 
(2) 寻找 一 个 下 标 i, 它 满足 条 件 : Finish[i]==false 且 Request[i, * ] 志 Work[ * ], 如 


果 找 不 到 这 样 的 i, 则 转向 (4)。 
(3) Work[ * ]= Work[ * j++ Allocation[i, * ]; Finish[ 站 =true; 转向 (2)。 


(4) 如 果 存 在 i(1 志 i 过 nn) ,Finish[ 让 一 false, 则 系统 处 于 死 锁 状态 。 若 Finish[ 让 一 


false, 则 进程 P; 处 于 死 锁 环 中 。 


在 上 面 的 算法 中 ,如 果 一 个 进程 所 申请 的 资源 能 够 满足 ,就 假定 该 进程 能 得 到 所 需 资 
源 ,向 前 推进 ,直至 结束 ,释放 所 占有 的 全 部 资源 。 接 着 查找 是 否 有 另外 的 进程 也 满足 这 种 


条 件 。 如 果 某 进程 在 以 后 还 要 不 断 申请 资源 , 则 它 还 可 能 会 被 检测 出 死 锁 。 


例如 , 设 系统 中 有 三 类 资源 {Ri,Rz,Rs} 和 5 个 并 发 执行 进程 {Pi ,Pz,P; ,P,Ps}) ,其 
中 ,Ri 有 7 个 ,Re 有 3 个 ,Rs 有 6 个 。 在 Te 时 刻 各 进程 分 配 资 源 和 申请 情况 如 表 8-1 


所 示 。 
表 8-1 5S 个 并 发 进程 的 资源 分 配 情况 
Allocation Request Available 
Ri R。 到 R， Ry Ra Ri R; Rs 
P， 0 1 0 0 0 0 
P: 2 0 0 2 0 2 
P; 3 0 3 0 0 0 0 1 0 
P, 2 1 . 1 0 0 
Ps 0 0 多 0 0 2 
此 时 ,系统 不 处 于 死 锁 状 态 , 运 行 上 述 死 锁 检测 算法 可 以 得 到 一 个 进程 序列 (Pi ,Ps:， 
P,P ,Ps), 对 于 所 有 的 i 都 有 Finish[i]==true, 如 表 8-2 所 示 。 
假定 进程 P; 现在 申请 一 个 单位 的 Rs 资源 , 则 系统 资源 分 配 情况 如 表 8-3 所 示 , 此 时 ， 


系统 处 于 死 锁 状态 ,参与 死 锁 的 进程 集合 为 {P: ,Ps ,Ps ,Ps) ,如 表 8-4 所 示 。 
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表 8-2 死 锁 检 测算 法 得 到 的 进程 序列 


Work Allocation Request Work 十 Allocation i 
Ri R; Ra Ri R; Rs Ri R2 Rs Ri R;: Ra 
Pi 0 | 0 0 1 0 0 0 0 0 2 0 true 
P: 0 2 0 3 0 3 0 0 0 3 2 3 true 
P: 3 多 3 2 0 0 2 0 2 2 3 true 
B 5 区 3 2 1 1 0 0 7 3 4 true 
Ps 7 3 4 0 0 发 0 0 2 7 3 6 true 
表 8-3 满足 进程 Ps 申请 后 的 资源 分 配 情 况 
Allocation Request Available 
Ri R;» R; Ri R; Ra Ri R;» Rs 
Pi 0 1 0 0 0 0 
P; 2 0 0 2 0 2 
Ps 3 0 3 0 0 | 0 1 0 
BR 2 1 1 1 0 0 
Ps 0 0 2 0 0 2 
表 8-4 死 锁 检测 算法 得 到 的 进程 序列 
Work Allocation Request Work 十 Allocation i 
Ri R; Rs: Ri R; R; Ri R; R; Ri R; Rs 
Pi 0 和 0 0 1 0 0 0 0 0 2 0 true 
系统 何 时 进行 死 锁 检测 呢 ? 这 取决 于 死 锁 出 现 的 频 度 和 当 死 锁 出 现时 影响 进程 数量 等 


因素 。 若 死 锁 经 常 出 现 , 检 测算 法 应 经 常 被 调用 。 一 科 


常用 的 方法 是 当 进 程 申 请 资源 不 能 


满足 就 进行 检测 。 死 锁 检 测 过 于 频繁 ,系统 开销 大 ; 如 检测 时 间 间 隔 过 长 , 卷 入 死 锁 的 进程 
会 增多 ,使 得 系统 资源 及 CPU 利用 率 大 为 下 降 , 一 个 折 中 的 办 法 就 是 定期 检测 ,如 每 一 
小 时 检测 一 次 ,或 在 CPU 的 利用 率 低 于 40% 时 检测 。 


8.2.2 从 死 锁 中 恢复 
当 发 现 有 进程 死 锁 时 , 便 应 立即 把 它们 从 死 锁 状态 中 恢复 出 来 。 常 采用 恢复 死 锁 的 两 


种 措施 是 : 一 是 通知 操作 员 死 锁 发 生 , 以 便 操作 人 员 人 工 处 理 现 


从 死 锁 状态 中 自动 恢复 过 来 。 死 锁 状 态 的 恢复 常 采用 下 述 两 种 方法 。 
1. 撤销 进程 


解除 死 锁 最 直接 的 方法 是 终止 一 个 或 若干 个 进程 ,系统 会 回 
有 资源 。 撤 销 进程 可 以 按照 某 种 顺序 逐步 撤销 已 死 锁 的 进程 , 直 型 
的 足够 可 用 的 资源 为 止 。 在 极端 情况 下 ,这 种 方法 可 能 造成 除 一 个 


进程 全 部 被 撤销 。 


在 撤销 进程 中 


收 分 配给 被 终 4 


获得 为 解除 珊 
死 锁 进程 外 ,其 余 的 死 锁 


需要 考虑 按照 什么 原则 撤销 进程 ,目前 比较 实 上 


E 锁 ; 另 一 种 方法 是 让 系统 


上 进程 的 所 
E 锁 所 需要 


而 简单 的 方法 是 撤销 那 
些 代价 最 小 的 进程 ,或 者 使 撤销 进程 的 数目 最 小 ,以 下 几 点 作为 衡量 撤销 代价 的 标准 。 


(1) 程序 的 优先 级 , 即 被 撤销 进程 的 优先 级 。 

(2) 作业 类 的 外 部 代价 ,规定 各 类 作业 各 自 的 撤销 代价 ,系统 可 以 根据 这 些 规定 撤销 代 
价 最 小 的 继承 ,达到 解除 死 锁 的 目的 。 

(3) 运行 代价 , 即 重新 启动 它 并 运行 到 当前 撤销 点 所 需要 的 代价 ,可 由 系统 记 账 程序 
给 出 。 

2. 和 剥夺 资源 

由 于 死 锁 是 由 进程 竞争 资源 而 引起 的 ,所 以 可 通过 剥夺 资源 以 取消 死 锁 ,逐步 从 进程 中 
剥夺 资源 给 其 他 进程 使 用 ,直到 死 锁 环 路 被 破坏 为 止 。 剥 夺 的 顺序 可 以 是 以 花费 最 小 资源 
数 为 依据 。 每 次 剥夺 后 ,需要 再 次 调用 检测 算法 。 资 源 被 剥夺 的 进程 为 了 再 次 得 到 该 资源 ， 
必须 重新 申请 。 

利用 剥夺 资源 的 方法 处 理 死 锁 ,需要 考虑 以 下 几 点 。 

(1) 选择 剥夺 哪些 进程 的 哪些 资源 。 与 撤销 进程 相同 ,必须 确定 剥夺 顺序 确保 代价 最 
小 化 。 代 价 因素 包括 进程 所 拥有 的 资源 数量 , 死 锁 进程 到 现在 为 止 在 执行 过 程 中 所 消耗 的 
时 间 等 。 

(2) 对 被 剥夺 资源 的 进程 的 安排 。 显 然 被 剥夺 资源 的 进程 缺少 所 需要 的 资源 ,不 能 正 
常 执行 。 建 立 检查 点 ,必须 将 进程 回 到 某 个 安全 状态 ,以便 从 该 状态 重启 进程 。 

(3) 保证 资源 不 会 总 是 从 同一 个 进程 中 被 剥夺 。 这 就 需要 确保 一 个 进程 只 能 有 限 次 地 
剥夺 资源 ,最 常用 的 方法 是 在 代价 因素 中 加 上 回 滚 次 数 。 


8.3 死 锁 避免 


死 锁 避免 的 方法 是 控制 系统 状态 的 变化 ,在 满足 资源 请 求 条 件 下 系统 做 出 判断 确定 不 
会 有 死 锁 发 生 时 才 会 把 资源 真正 分 配给 进程 。 因 此 首先 需要 分 析 预 期 的 状态 , 即 在 进入 状 
态 之 前 ,要 分 析 是 否 存在 某 一 状态 变换 序列 ,能 从 进入 的 状态 中 安全 退出 ,保证 其 中 每 个 进 
程 都 可 以 执行 。 下 面 将 首先 对 状态 进行 介绍 ,然后 介绍 一 个 死 锁 避 免 的 具体 方法 , 即 银行 家 
算法 。 


8.3.1 安全 状态 与 不 安全 状态 


所 谓 安 全 状态 是 指 如 果 系统 能 按照 某 个 顺序 ,如 < Pi,P:,…',P, >, 为 每 个 进程 分 配 资 
源 ( 不 超过 其 最 大 值 ) 使 得 每 个 进程 都 能 执行 完毕 ,此 时 称 系统 处 于 安全 状态 。 进 程 执行 序 
列 < Pi ,P:,…,P, > 为 当前 系统 的 一 个 安全 序列 ,安全 序列 是 不 唯一 的 。 具 体 地 说 ,在 当前 分 
配 状态 下 ,进程 的 安全 序列 < Pi ,P:,…',P, > 是 这 样 组 成 的 : 若 对 于 每 一 个 进程 Pi(1<i<n)， 
它 需要 的 附加 资源 可 被 系统 中 当前 可 用 资源 与 所 有 进程 PC 到 i) 当 前 占有 资源 之 和 所 满 
足 , 则 < Pi ,Ps ,…,P, > 为 一 个 安全 序列 。 如 果 不 存在 这 样 的 顺序 ,那么 系统 状态 就 处 于 不 
安全 状态 。 

由 安全 状态 定义 可 知 ,处 于 安全 状态 的 系统 一 定 不 会 产生 死 锁 。 即 使 某 个 进程 P; 所 需 
的 资源 总 数 超过 系统 当前 可 用 空 闭 的 资源 总 数 ,P; 可 以 等 待 其 他 进程 执行 完毕 后 释放 资 
源 ,系统 把 释放 的 资源 分 配给 进程 P;,,P; 最 终 能 获得 所 需 的 资源 ,从 而 执行 完毕 。 
并 非 所 有 的 不 安全 状态 都 会 导致 死 锁 状态 。 但 当 系 统 进入 不 安全 状态 时 , 便 有 了 导致 


死 锁 


攻 o 址 


所 作 系 统 原 理 


死 锁 状态 的 可 能 。 如 果 能 保证 每 次 资源 分 配 后 ,系统 都 处 于 安全 状态 , 则 不 会 发 生死 锁 。 

例如 ,考虑 某 一 系统 ,共有 10 台 打 印 机 ,在 Te 时刻 三 个 进程 Pi,Ps,Ps 分 别 最 多 需要 8 
台 、7 台 和 4 台 打 印 机 。 假 定 Pi ,P,Ps 已 分 别 申请 到 4 台 、 两 台 和 两 台 打 印 机 。 当 前 系统 
情况 如 表 8-5 所 示 。 


表 8-5 Te 时 刻 系统 当前 状态 


最 大 需求 当前 已 分 配 当前 需求 
P， 8 4 4 
P: 5 2 5 
Ps 4 用 加 


由 于 系统 共有 10 台 打 印 机 ,To 时 刻 Pi ,P,Ps 已 申请 到 4 台 、 两 台 和 两 台 打印 机 ,因此 
系统 中 还 有 两 台 空 闲 打印 机 。 进 程 Ps 可 立即 得 到 需要 的 所 有 打印 机 并 在 完成 工作 后 归还 
它们 (系统 会 有 4 台 打印 机 空闲 ), 接 着 进程 Pi 可 得 到 需要 的 所 有 打印 机 并 在 完成 工作 后 归 
还 它们 (系统 会 有 8 台 打 印 机 空闲 ) ,最 后 进程 Ps 得 到 需要 的 打印 机 并 执行 完成 后 归还 它们 
(系统 会 有 10 台 打 印 机 空间 )。 所 以 可 以 找到 一 个 安全 序列 < Ps,Pi,P: >, 因 此 在 Te 时 刻 
系统 处 于 安全 状态 。 

自 定 在 时 刻 Ti ,进程 P 申请 并 又 得 到 一 台 打 印 机 ,那么 系统 状态 见 表 8-6。 此 时 系 
统 处 于 不 安全 状态 。 系 统 中 只 有 一 台 空闲 打印 机 可 用 ,这 一 台 打 印 机 无 法 满足 任何 一 个 
进程 的 请 求 ,因此 也 无 法 找到 一 个 安全 序列 ,可 见 系统 是 有 可 能 从 安全 状态 转 到 不 安全 
状态 的 。 


表 8-6 Ti 时刻 系统 当前 状态 


最 大 需求 当前 已 分 配 当前 需求 
I 8 4 4 
P， 8 a 4 
Ps 4 2 2 


从 以 上 介绍 可 以 看 出 以 下 几 点 。 

(1) 死 锁 状 态 是 不 安全 状态 。 

(2) 如 果 处 于 不 安全 状态 ,并 不 意味 着 它 就 在 死 锁 状 态 , 而 是 表示 存在 导致 死 锁 的 
危险 。 

(3) 如 果 一 个 进程 申请 的 资源 当前 是 可 用 的 ,但 采用 死 锁 避 免 的 方法 ,该 进程 也 可 能 需 
要 等 待 ,这 和 进程 的 检查 与 恢复 比 ,资源 的 利用 率 会 有 所 下 降 。 
8.3.2 银行 家 算法 

最 著名 的 死 锁 避免 的 算法 叫 作 “银行 家 算法 ”, 是 由 Dijkstra 首先 提出 并 加 以 解决 的 。 
该 算法 如 此 命名 是 因为 它 可 用 于 银行 系统 。 假 定 小 镇 银行 家 拥有 资金 ,被 多 个 客户 共享 , 银 
行家 对 客户 提出 以 下 约束 条 件 。 

(1) 每 个 客户 必须 预先 说 明 所 要 的 最 大 资金 量 ; 

(2) 每 个 客户 每 次 提出 部 分 资金 量 的 请 求 并 获得 分 配 ; 


(3) 如 果 银 行 满足 客户 对 资金 的 最 大 需求 量 , 那 么 客户 在 资金 运作 后 ,应 在 有 限 的 时 间 
内 全 部 归还 银行 。 

只 要 客户 遵守 上 述 约束 条 件 , 银 行家 保证 如 果 客 户 所 要 的 最 大 资金 量 不 超过 银行 家 现 
有 资金 ,银行 一 定 会 接纳 该 客户 ,并 满足 其 资金 需求 ; 银行 家 在 收 到 一 个 客户 的 资金 申请 
后 ,可 能 会 因 资金 不 足 或 者 如 果 把 资金 分 配给 申请 的 客户 后 ,导致 所 有 客户 都 不 能 将 资金 归 
还 ,那么 银行 家 不 会 满足 客户 的 申请 ,但 保证 在 有 限 的 时 间 内 让 客户 获得 资金 。 在 银行 家 算 
法 中 ,客户 可 看 作 进程 ,资金 可 看 作 资 源 ,银行 家 可 看 作 操 作 系统 。 银 行家 算法 虽然 能 避免 
死 锁 ,但 实现 时 受到 种 种 限制 ,要 求 所 涉及 的 进程 不 相交 , 即 不 能 有 同步 要 求 ,事先 要 知道 进 
程 的 总 数 和 每 个 进程 请 求 的 最 大 资源 数 ,这 些 都 很 难 办 到 。 

在 银行 家 的 例子 中 我 们 看 到 资源 只 有 一 个 ,其 实 银行 家 算法 可 以 处 理 多 个 资源 。 为 了 
实现 银行 家 算法 , 先 介 绍 几 个 数据 结构 ,这 些 数 据 结构 对 资源 分 配 系 统 的 状态 进行 了 记录 。 
设 n 为 系统 进程 的 个 数 ,m 为 资源 类 型 的 种 类 。 需 要 如 下 数据 结构 。 

(1) 当前 可 分 配 的 空闲 资源 向 量 Available, 其 为 长 度 为 m 的 向 量 。 如 果 Available[ 庄 一 
k, 那 么 表示 系统 中 Ri 类 型 资源 可 用 的 数量 是 A。 

(2) 最 大 需求 矩阵 Max, 其 为 nXm 矩阵 。Max[Li, 门 一 & 表 示 进 程 P; 至 多 可 申请 个 
R; 类 资源 。 

(3) 资源 分 配 和 矩阵 Allocation, 其 为 nXm 和 矩阵 。Allocation[i,j==k 表示 进程 P; 当前 
分 到 k 个 R; 类 资源 。 

(4) 需求 矩阵 Need, 其 为 ?2X 痉 矩阵 。Need[Li, 门 表示 进程 P; 尚 需 k 个 Ri 类 资源 才能 
完成 其 任务 。 

上 述 三 个 矩阵 的 关系 为 Need[i,j]= 二 Max[i,jj] 一 Allocation[i,j]。 

为 了 简化 起 见 , 可 以 把 矩阵 Allocation 和 Need 矩阵 中 的 每 一 行当 中 一 个 向 量 ,针对 进 
程 Pi 已 分 配 和 还 需要 的 向 量 分 别 写成 Allocation 和 Needi。 

设 Request 为 进程 P; 的 请 求 向 量 , 如 果 Requesti[ 门 =&, 那 么 进程 P; 申请 & 个 R 类 
资源 。 银 行家 算法 如 下 。 

(1) 申请 量 超过 最 大 需求 量 时 出 错 , 否 则 转 步 骤 (2)。 

(2) 当 申 请 量 超过 当前 系统 所 拥有 的 可 分 配 量 时 , 挂 起 进程 ,该 进程 处 于 阻塞 态 ,否则 
转 步 骤 (3) 。 

(3) 系统 对 进程 P; 请 求 的 资源 进行 试探 性 分 配 ,执行 

Allocation[ i, * ] = Allocation[ i, * ] +Requesti[*] 
Available[ * ] = Available[ * ] -Requesti[ *] 
Need[ i, * ] = Need[ i, * ] ~ Requesti:[ *] 

(4) 执行 安全 性 测试 算法 (5) ,如 果 状 态 安全 则 承认 该 试探 性 分 配 , 否 则 抛弃 该 试探 性 
分 配 ,进程 P; 阻塞 。 

(5) 安全 性 测试 算法 。 

@ 定义 工作 向 量 Work 和 布尔 型 标志 Finish, 即 Work 和 Finish 分 别 是 长 度 为 m 和 nn 
的 向 量 。 按 照 如 下 方式 进行 初始 化 : Work[ * ] 二 Available[ * ],FinishL * ] 二 false。 

@ 查找 这 样 的 i, 使 其 满足 Finish[i] 二 false, 并 且 Need[i,x ] 三 Work[ * ]。 如 果 没 有 
这 样 的 i 存在 , 转 步 骤 @。 
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@ 执行 
Work[ * ] = Work[ * ] + Allocation[i,*] 
Finish[i] = true 
返回 步骤 @ 。 
@ 如 果 对 于 所 有 的 i,Finish[L 让 一 true, 那 么 系统 处 于 安全 状态 ,返回 安全 标记 ,和 否则 返 
可 不 安全 标记 。 
下 面 从 单 资源 到 多 资源 ,由 浅 到 深 举 例 来 说 明 银 行家 算法 。 
例 8-1 系统 中 有 一 种 资源 总 量 为 10。 三 个 进程 A,B,C 的 最 大 资源 需求 量 分 别 是 9， 
4,7, 目 前 已 得 到 的 分 配 量 是 3,2,2, 如 表 8-7 所 示 。 
表 8-7 系统 资源 分 配 情况 


进程 最 大 需求 已 有 分 配 量 
A 9 3 
B 4 2 
C 7 2 


(1) 此 状态 是 否 是 安全 状态 ? 

(2) 现在 进程 B 提 出 一 个 资源 请 求 ,系统 应 该 接受 该 请 求 吗 ? 

(3) 在 表 8-7 的 基础 上 进程 A 提出 一 个 请 求 ,系统 应 该 接受 该 请 求 吗 ? 
解 : 

(1) 根据 题 意 ,系统 的 资源 分 配 情况 如 表 8-8 所 示 。 


表 8-8 (1) 的 系统 资源 分 配 情况 


进程 Max Allocation Need Available 
A 人 3 6 
B 4 2 2 3 
€ 7 2 5 


此 时 存在 一 个 安全 序列 < B,C,A >, 系 统 处 于 安全 状态 ,如 表 8-9 所 示 。 
表 8-9 安全 状态 分 析 


进程 Work Need Allocation Work 十 Allocation Finish 
B 3 2 2 5 true 
C 5 5 2 7 true 
A 过 6 3 10 true 


(2) 当 B 提 出 一 个 资源 请 求 .其 提出 的 请 求 既 没 超 过 自己 的 最 大 需求 量 ,也 没有 超过 可 
用 资源 数量 , 即 
Requests(1)< Needs(2) 
Requests(1)<Available(3) 
系统 假定 接受 它 , 资 源 分 配 如 表 8-10 所 示 。 


表 8-10 (2) 的 系统 资源 分 配 情况 


进程 Max Allocation Need Available 
A 9 3 6 
B 4 1 


析 表 。 
表 8-11 B 申请 资源 时 的 安全 状态 分 析 
进程 Work Need Allocation Work 十 Allocation Finish 
也 2 1 3 5 true 
We 5 5 2 § true 
A * 6 3 10 true 


此 时 存在 一 个 安全 序列 < B,C,A >, 系 统 处 于 安全 状态 ,可 以 正式 把 资源 分 配给 B。 
(3) 当 A 提出 一 个 资源 请 求 ,其 提出 的 请 求 既 没 超过 自己 的 最 大 需求 量 , 也 没有 超过 可 
用 资源 数量 , 即 


Requesta(1)<Needa (6) 
Requesta(1)<Available(3) 
系统 假定 接受 它 ,资源 分 配 如 表 8-12 所 示 。 
表 8-12 (3) 的 系统 资源 分 配 情况 


进程 Max Allocation Need Available 


再 用 安全 性 测试 算法 检查 系统 此 时 的 状态 是 否 安全 ,得 到 如 表 8-13 所 示 的 安全 状态 分 
析 表 。 


表 8-13 A 申请 资源 时 的 安全 状态 分 析 


进程 Work Need Allocation Work 十 Allocation Finish 


B 2 2 2 4 true 


此 时 系统 不 存在 一 个 安全 序列 ,系统 处 于 不 安全 状态 ,在 进程 B 执行 完 后 ,系统 无 法 满 
足 其 余 进程 的 需求 ,不 能 为 A 分 配 资源 。 

例 8-2 假设 系统 中 共有 5 个 进程 {Po ,Pi ,Ps,P;,P1} 和 A,B,C 三 类 资源 ; A 类 资源 共 
有 10 个,B 类 资源 共有 5 个,C 类 资源 共有 7 个。 在 时 刻 T,, 系统 资源 分 配 情况 如 表 8-14 
所 示 。 
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表 8-14 系统 资源 分 配 情况 


Allocation Max Available 
进程 
A B C A B G A B © 
Ps 0 1 0 7 5 3 
P， 2 0 0 3 2 区 
Ps 3 0 4 9 0 2 3 3 2 
Ps 2 0 1 2 2 
Ps 0 0 2 4 3 3 
(1) To 时 刻 是 否 安全 。 


(2) Pi 又 要 请 求 一 个 A 类 资源 和 两 个 C 类 资源 ,系统 应 该 接受 该 请 求 吗 ? 
(3) 在 (2) 的 基础 上 ,Ps 请 求 三 个 A 类 资源 和 三 个 B 类 资源 ,系统 应 该 接受 该 请 求 吗 ? 
(4) 在 (2) 的 基础 上 ,P。 请 求 两 个 B 类 资源 ,系统 应 该 接受 该 请 求 吗 ? 
解 : 
(1) 根据 题 意 ,系统 的 资源 分 配 情况 如 表 8-15 所 示 。 
表 8-15 (1) 的 系统 资源 分 配 情况 


Allocation Max Need Available 
进程 
A B C A B C A B C A B C 
Pp, 0 1 0 各 5 3 7 4 3 
孜 2 0 0 3 2 2 1 2 2 
P; 3 0 4 9 0 2 6 0 0 3 3 2 
Ps 2 1 1 2 2 2 0 1 1 
Pp, 0 0 2 1 3 3 4 3 1 


此 时 存在 一 个 安全 序列 < Pi ,P: ,P,,P:,P。 >, 系 统 处 于 安全 状态 ,如 表 8-16 所 示 。 
表 8-16 ”安全 状态 分 析 


资源 Work Need Allocation Work 十 Allocation a 
进程 入 攻破 6 A B C 和 
Pi 3 3 2|1 2 2|2 0 0 5 3 2 true 
Ps: 5 3 2|10 1 1 2 1 1 (8 4 2 true 
By 7 4 3|I4 3 1|0 0 2 区 14 5 true 
P， 74 $16 0 |I3 0 2 10 4 7 true 
Po 现 击 证 | 可 4 3|0 1 0 10 5 7 true 


(2) 按照 银行 家 算法 进行 检测 : 
Request1(1,0,2)<Needi(1.2,2) 
Request1(1.,0,2)<Available(3.3.2) 
系统 尝试 先 为 Pi 分 配 资源 ,修改 Allocation,Need 和 Available, 得 到 如 表 8-17 所 示 的 
新 状态 。 


表 8-17 (2) 的 系统 资源 分 配 情况 


Allocation Max Need Available 
进程 
改 家 区 | 如 已 | 入 村 已 A B ( 
Bs 0 i 0 7 5 3 7 4 3 
Pi 3 洲 ” 元 8 2 2 | 0 名 0 
P; 3 0 2 9 0 2 6 0 0 2 3 0 
2 1 1 2 2 受 ‖ 6 1 1 
P, 0 0 2 1 3 3 4 3 1 
再 用 安全 性 测试 算法 检查 系统 此 时 的 状态 是 否 安全 ,得 到 如 表 8-18 所 示 的 安全 状态 分 
析 表 。 
表 8-18 P, 申请 资源 时 的 安全 状态 分 析 


Allocation Work 十 Allocation 
和 Finish 
进程 A BB CIA BB CIA BC A B C 
Pi 2 3 0 0 2 0 3 0 到 5 3 2 true 
P; 5 和 和 | 吕 :和 人 7 4 2 true 
P, 7 1 3 1 3 1 0 0 | 7 4 5 true 
Po 4? 1 5 7 1 3 0 1 0 区 5 5 true 
true 


可 见 能 够 找到 一 个 安全 序列 ~ Pi ,Ps ,P, ,Po,P: >, 因 此 ,系统 此 时 处 于 安全 状态 ,可 以 


把 资源 分 配给 Pl。 
(3) 按照 银行 家 算法 进行 检测 : 
Requests4(3,3,0)<Needs(4,3,1) 


Requests(3,3,0) 三 Available(2,3,0) 不 成 立 
由 于 这 时 可 用 系统 资源 不 足 , 申 请 被 系统 拒绝 , 令 进程 P, 阻塞 。 


(4) 按照 银行 家 算法 进行 检测 : 
Requesto(0,2,0)<Needo(7,4.,3) 
Requesto(0,2,0)<Available(2,3.,0) 

系统 尝试 先 为 Po 分 配 资源 ,修改 Allocation,Need 和 Availab 

表 8-19 (4) 的 系统 资源 分 配 情况 


e, 得 到 如 表 8-19 所 示 的 


Allocation Max Need Available 
进程 
A B C A B C A B € A B C 
Po 0 3 0 要 5 3 芝 要 3 
P， 3 0 2 3 2 2 0 2 0 
Pp, 0 2 9 0 2 6 0 0 2 1 0 
Ps 2 1 1 2 2 2 0 1 | 
Ps 0 0 2 4 3 3 4 3 
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此 时 ,利用 安全 性 测试 算法 检查 发 现 ,剩余 资源 已 不 能 满足 任何 进程 的 新 需求 , 故 系统 
已 处 于 不 安全 状态 ,不 能 为 Po 分 配 资源 。 


8.4 死 锁 预防 


前 面 介绍 了 出 现 死 锁 的 情况 有 4 个 必要 条 件 ,如 果 能 对 进程 施加 适当 的 限制 ,只 要 确保 
至 少 一 个 必要 条 件 不 成 立 , 就 能 预防 死 锁 的 发 生 。 


8.4.1 破坏 互 斥 


如 果 人 允许 系统 中 的 所 有 资源 都 能 共享 使 用 , 即 破 坏死 锁 的 “ 互 斥 ”使 用 条 件 , 系 统 将 不 会 
发 生死 锁 。 但 是 ,临界 资源 本 身 的 性 质 决 定 其 不 能 共享 使 用 ,否则 无 法 保证 正确 性 。 打印 机 
就 是 一 个 典型 代表 ,如 进程 能 共享 使 用 打印 机 , 则 打印 出 的 结果 没有 任何 意义 。 系 统 可 以 通 
过 借助 SPOOLing 技术 允许 多 个 进程 同时 产生 打印 数据 。 该 模型 中 唯一 真正 申请 物理 打 
印 机 的 进程 是 打印 管理 进程 ,由 于 它 决 不 会 申请 别 的 设备 ,所 以 不 会 因 打印 机 而 发 生死 锁 ， 
但 SPOOLing 技术 并 非 适 用 于 所 有 的 设备 (如 进程 表 ), 而 且 在 SPOOLing 对 磁盘 空间 本 身 
进行 竞争 时 也 可 能 导致 死 锁 。 所 以 这 种 做 法 在 许多 场合 行 不 通 。 系 统 不 但 不 能 破坏 * 互 斥 ” 
条 件 , 还 要 采取 各 种 办 法 保证 独占 资源 的 互 斥 使 用 。 

因此 , 死 锁 预防 策略 主要 集中 针对 其 他 三 个 条 件 来 进行 。 


8.4.2 破坏 占有 且 等 待 


破坏 这 个 条 件 的 办 法 很 简单 ,可 采用 静态 分 配 资源 方法 。 所 谓 静 态 分 配 策略 也 称 为 次 
源 的 一 次 性 分 配 法。 进程 运行 前 ,系统 一 次 性 地 分 配 其 运行 所 需 的 所 有 资源 ,而 不 是 随 着 进 
程 的 推进 陆续 分 配 。 由 于 进程 运行 前 一 次 性 地 获得 了 所 需 的 所 有 资源 ,该 进程 可 顺利 执行 
完毕 ,不 会 发 生死 锁 。 

这 种 预防 死 锁 的 方法 的 优点 是 简单 .易于 实现 且 很 安全 。 但 这 种 策略 严重 地 降低 了 资 
源 利用 率 。 因 为 ,在 每 个 进程 所 占有 的 资源 中 ,有 些 资源 在 进程 执行 后 期 才 使 用 ,甚至 有 些 
资源 在 例外 的 情况 下 被 使 用 。 因 此 就 可 能 造成 一 个 进程 占有 了 一 些 几 乎 不 用 的 资源 而 使 其 
他 想 用 这 些 资 源 的 进程 产生 等 待 。 

例如 ,一 个 进程 需要 将 数据 从 DVD 驱动 器 复制 到 磁盘 文件 ,并 对 磁盘 文件 进行 排序 ， 
再 将 结果 打印 。 如 果 所 有 资源 必须 在 进程 运行 之 前 申请 ,那么 进程 必须 一 开始 就 申请 DVD 
了 驱动 器 ,磁盘 文件 和 打印 机 。 在 整个 执行 期 间 , 进 程 会 一 直 占 用 驱动 器 ,磁盘 文件 和 打印 机 ， 
直到 运行 结束 才 释 放 。 打 印 机 是 在 进程 运行 结束 时 才 完 成 打印 工作 的 ,因此 该 设备 在 整个 
分 配 过 程 中 很 长 时 期 都 处 于 空闲 状态 。 

8.4.3 破坏 不 可 抢占 

抢占 调度 能 够 防止 死 锁 ,但 一 般 只 适用 于 主 存 和 处 理 器 资源 。 方 法 一 是 占有 资源 的 进 
程 车 要 申请 新 的 资源 ,必须 主动 放弃 已 占用 资源 (抢占 式 ), 若 仍 需 占有 此 资源 ,应 该 向 系统 
重新 提出 申请 ,从 而 破坏 了 不 抢占 条 件 , 但 会 造成 进程 重复 的 申请 和 释放 资源 ,一 般 不 采 
此 方法 ; 方法 二 是 资源 管理 程序 为 进程 分 配 新 资源 时 , 若 有 则 分 配 之 ,否则 将 抢占 此 进 


程 所 占有 的 全 部 资源 ,并 让 进程 进入 等 待 资源 的 状态 ,资源 充足 后 再 唤醒 它 重新 申请 所 

破坏 不 可 抢占 实现 复杂 ,开销 较 大 。 因 为 一 个 资源 在 使 用 一 段 时 间 后 , 它 的 被 迫 释 放 可 
能 会 造成 前 段 时 间 工 作 的 失效 ,即使 采取 了 某 些 防范 措施 ,也 还 会 使 进程 前 后 两 次 运行 的 信 
息 不 连续 。 例 如 ,进程 在 运行 过 程 中 已 用 打印 机 输出 信息 ,但 中 途 又 因 申请 另 一 资源 未 果 而 
被 迫 暂停 运行 并 释放 打印 机 ,后 来 系统 又 把 打印 机 分 配给 其 他 进程 使 用 。 当 进程 再 次 恢复 
运行 并 再 次 获得 打印 机 继续 打印 时 ,这 前 后 两 次 打印 输出 的 数据 并 不 连续 , 即 打印 输出 的 信 
息 中 间 有 一 段 是 另 一 进程 的 。 此 外 ,该 策略 还 可 能 由 于 反复 地 申请 和 释放 资源 ,使 进程 的 执 
行 无 限 推迟 ,延长 了 进程 的 周转 时 间 ,增加 了 系统 开销 ,降低 了 系统 吞吐 量 。 因 此 可 以 看 出 
破坏 不 可 抢占 的 主要 缺点 是 系统 代价 较 大 , 即 在 剥夺 用 户 资源 时 ,要 保存 进程 的 上 下 文 现 
场 , 在 进程 重新 获得 资源 时 ,要 恢复 进程 上 下 文 现场 ,系统 开销 较 大 ,程序 执行 延迟 。 同 时 临 
界 资源 不 可 抢占 。 如 果 强 行 抢占 ,会 产生 不 良 后 果 。 


8.4.4 破坏 环 路 等 待 


环 路 等 待 出 现 的 根本 原因 是 并 发 执行 进程 请 求 资源 的 顺序 是 随机 的 。 假 如 有 两 个 进 
程 ,一 个 进程 先 申请 资源 A 再 申请 资源 B, 另 一 个 进程 先 申请 资源 B 再 申请 资源 A。 这 两 
个 进程 随机 推进 时 ,就 可 能 产生 死 锁 。 如 果 系 统 事先 对 所 有 资源 类 型 进行 排序 , 且 要 求 每 个 
进程 按照 递增 顺序 来 申请 资源 ,这 样 一 来 ,进程 在 申请 、 占 用 资源 时 就 不 会 形成 资源 申请 环 
路 ,也 就 不 会 发 生 环 路 等 待 。 

例如 , 某 系统 规定 磁带 机 的 整数 编号 为 1 ,磁盘 机 的 编号 为 6, 打印 机 的 编号 为 16。 按 
照 上 述 规定 ,一 个 进程 在 执行 过 程 中 要 使 用 磁带 机 和 打印 机 ,进程 必须 先 申 请 磁带 机 ,然后 
再 申请 打印 机 ,不 能 颠倒 次 序 , 就 可 以 防止 系统 发 生死 锁 , 因 为 采用 有 序 使 用 资源 的 方法 能 
够 有 效 地 破坏 环 路 等 待 条 件 。 

一 般 情 况 下 ,系统 给 资源 进行 编号 时 要 充分 考虑 资源 的 一 般 使 用 顺序 。 例 如 ,输入 设备 
通常 在 输出 设备 之 前 被 使 用 ,因此 磁盘 机 的 编号 通常 比 打印 机 的 编号 小 。 

将 系统 中 所 有 资源 按 类 型 编号 、 进 程 按照 资源 编号 递增 顺序 申请 资源 的 办 法 消除 了 死 
锁 产 生 的 必要 条 件 ,同时 提高 了 系统 资源 利用 率 和 系统 吞吐 量 。 但 此 方法 也 存在 着 不 足 ,一 
方面 用 户 进程 必须 知道 资源 的 请 求 顺序 ,不 方便 用 户 使 用 ; 另 一 方面 ,如 果 进程 当前 不 使 用 
某 个 资源 Ri, 但 是 由 于 要 请 求 序 号 比 资源 Ri 大 的 资源 Ri 而 要 先 请 求 资源 Ri ,这 就 会 导致 
资源 Ri 分 配给 进程 后 要 隔 很 久 才 会 被 用 到 ,从 而 导致 资源 Ri 利用 率 的 降低 。 此 外 ,给 系统 
中 所 有 资源 合理 的 编号 也 非 易 事 。 资 源 编号 一 旦 确定 就 须 相 对 稳定 ,这 在 某 种 程度 上 限制 
了 新 类 型 资源 的 增加 。 


8.5 活 锁 与 饥饿 


1. 活 锁 
解决 进程 互 斥 的 进入 临界 区 或 存 取 资源 时 可 以 利用 锁 原 语 实现 ,但 是 我 们 都 知道 这 种 
机 制 存 在 忙 等 问题 。 例 如 ,进程 A 和 B 都 要 使 用 临界 资源 R 和 S, 它 们 的 执行 流程 如 下 


死 争 


好 o 吉 


所 作 系 统 原 理 


进程 A 进程 B 

lock (R); lock (S); 

lock (S); lock (R); 

使 用 资源 R 和 5; 使 用 资源 R 和 5; 
释放 资源 R 和 5S; 释放 资源 R 和 5S; 
unlock (S); unlock (R); 
unlock (R); unlock (S); 


如 果 进 程 A 先 运行 并 得 到 资源 RR, 接 着 进程 运行 并 得 到 资源 S, 那 么 下 面 不 管 哪个 程 
序 运行 都 不 会 有 任何 进展 。 其 结果 是 : A 轮 询 , 消 耗 完 分 给 它 的 时 间 片 ,切换 到 B,B 轮 询 ， 
消耗 完 分 给 它 的 时 间 片 ,再 切换 到 A, 如 此 往复 。 从 表面 上 看 ,好 像 发 生死 锁 了 ,但 是 它们 都 
没有 被 阻塞 ,都 可 以 活动 ,这 种 现象 就 是 活 锁 。 

活 锁 也 经 常 出 人 意料 地 产生 ,可 以 通过 限制 进程 对 资源 的 申请 数量 或 规定 申请 顺序 等 
办 法 来 解决 。 其 实 比 起 限制 用 户 去 使 用 一 个 进程 .一 个 打开 的 文件 或 任意 一 种 资源 来 说 ,大 
多 数 用 户 可 能 更 愿意 选择 一 次 偶然 的 活 锁 ( 或 者 甚至 是 死 锁 ) 。 所 以 在 解决 活 锁 或 死 锁 的 过 
程 中 ,要 考虑 效率 和 便捷 性 等 问题 。 

2. 饥饿 

与 死 锁 和 活 锁 非 常 相似 的 一 个 问题 是 饥饿 。 进 程 在 其 生命 周期 中 需要 很 多 不 同类 型 的 
资源 。 由 于 进程 往往 是 动态 创建 的 .这 样 在 任何 时 候 系统 中 都 会 出 现 资源 申请 , 何 时 为 哪个 
进程 分 配 资源 ,以 及 分 配 多 少 资源 都 是 系统 分 配 资源 的 策略 问题 。 在 某 些 策略 下 ,系统 会 出 
现 这 样 一 种 状态 : 在 可 预计 的 时 间 内 , 某 个 或 某 些 进程 永远 得 不 到 完成 工作 的 机 会 ,因为 它 
们 所 需 的 资源 总 是 被 别 的 进程 占有 或 抢占 , 即 进程 被 无 限制 地 推 后 ,尽管 它 可 能 没有 被 阻 
塞 ,这 种 状态 称 作 “ 饥 饿 ”。 

例如 ,考虑 打印 机 分 配 。 现 在 假设 若干 个 进程 同时 请 求 打 印 机 ,究竟 哪 一 个 进程 能 获得 
打印 机 呢 ? 系统 可 能 采用 一 种 可 能 的 方案 : 优先 把 打印 机 分 配给 打印 的 文件 最 小 的 那个 
(假设 这 个 信息 可 知 )。 这 个 方法 让 尽量 多 的 用 户 满意 ,并 看 起 来 公平 ,但 存在 这 样 一 个 可 能 
性 : 在 一 个 繁忙 的 系统 中 , 某 个 进程 要 打印 的 文件 很 大 ,每 当 打印 机 空闲 ,系统 纵 观 所 有 进 
程 ,并 把 打印 机 分 配给 打印 最 小 文件 的 进程 。 如 果 在 一 个 稳定 的 进程 流 中 ,其 中 的 进程 都 是 
只 打印 小 文件 ,那么 要 打印 大 文件 的 进程 永远 也 得 不 到 打印 机 ,导致 它 会 饥饿 而 死 。 

饥饿 虽然 与 死 锁 相近 ,但 不 同 于 死 锁 , 死 锁 的 进程 都 必定 处 于 阻塞 状态 ,而 饥饿 的 进程 
不 一 定 被 阻塞 ,可 以 在 就 绪 状 态 。 

饥饿 可 以 通过 先 来 先 服务 资源 分 配 策略 来 避免 ,采用 这 种 方法 等 待 最 久 的 进程 可 以 成 
为 下 一 个 被 服务 的 进程 ,从 而 获得 所 需 资源 完成 自己 的 工作 。 


小 结 


死 锁 是 所 有 操作 系统 都 面临 的 潜在 问题 。 在 一 组 进程 中 ,每 个 进程 都 因 等 待 由 该 组 进 
程 中 的 另 一 进程 所 占有 的 资源 而 导致 阻塞 , 死 锁 就 发 生 了 。 死 锁 产 生 的 根本 原因 有 两 个 : 
一 是 系统 中 的 资源 数目 不 能 满足 多 个 并 发 进程 的 全 部 资源 需求 ,各 进程 竞争 资源 ,如 系统 对 
资源 分 配 不 合理 就 会 产生 死 锁 , 简 记 为 资源 竞争 : 二 是 并 发 执行 进程 间 的 推进 顺序 不 合理 
也 可 能 产生 死 锁 , 简 记 为 推进 顺序 非法 。 


系统 产生 死 锁 有 4 个 必要 条 件 : 互 斥 条 件 、 占 有 且 等 待 条 件 、 不 抢占 条 件 和 环 
件 。 解 决 死 锁 的 方法 有 : 死 锁 检测 和 恢复 、 避 免 死 锁 、 预 防 死 锁 。 死 锁 的 检测 和 恢 
资源 的 申请 和 分 配 不 施加 任何 限制 ,但 必须 建立 检测 机 制 ,周期 性 地 检测 是 否 发 4 


复 表示 对 


果 检 测 发 现 死 锁 则 采取 措施 恢复 死 锁 。 死 锁 的 避免 采用 动态 分 析 和 检测 新 的 资源 
源 的 分 配 情况 ,以 确保 系统 始终 处 于 安全 状态 ,其 中 最 著名 的 算法 是 银行 家 算法 。 
防 包括 一 切 都 是 用 假 脱 机 技术 (破坏 互 斥 条 件 ) ,资源 一 次 性 分 配 ( 破 坏 占有 且 等 
抢占 资源 (破坏 不 抢占 条 件 ); 资源 有 序 分 配 法 (破坏 环 路 等 待 条 件 ) 。 

活 锁 和 饥饿 是 同 死 锁 非常 相似 的 问题 ,但 在 技术 上 不 同 , 活 锁 包 含 的 是 实际 
住 的 进程 ,而 饥饿 可 以 通过 先 来 先 服 务 的 分 配 策 略 来 避免 。 


E 死 锁 , 如 
请 求 和 资 
死 锁 的 预 
待 条 件 ); 


没有 被 锁 


姓 o 直 


第 9 章 多 处 理 机 系统 介绍 


从 计算 机 诞生 之 日 起 ,人 们 对 更 强 计 算 能 力 的 无 休止 追求 就 一 直 驱 使 着 计算 机 工业 的 
发 展 。 如 今 普通 计算 机 的 运算 速度 已 达到 了 每 秒 几 十 亿 次 , 比 早期 计算 机 的 速度 快 了 百 万 
倍 , 但 是 还 有 对 更 强大 机 器 的 需求 。 生 物 学 家 正在 试图 揭 开 人 类 基因 的 奥秘 ,天 文学 家 正在 
了 解 宇宙 ,航空 工程 师 们 致力 于 建造 更 安全 和 速度 更 快 的 飞机 ,而 所 有 这 一 切 都 需要 更 快 的 
CPU。 然 而 ,即使 CPU 的 速度 不 断 提升 ,仍然 不 能 满足 需求 。 

过 去 的 解决 方案 是 使 时 钟 走 得 更 快 。 但 是 ,现在 开始 遇 到 对 时 钟 速度 的 限制 了 。 按 昭 
爱 因 斯 坦 的 相对 论 , 电 子 信号 的 速度 不 可 能 超过 光速 ,这 个 速度 在 真空 中 大 约 是 30cm/ns， 
而 在 铜 线 或 光纤 中 约 是 20cm/ns。 这 在 计算 机 中 意味 着 10GHz 的 时 钟 ,信号 的 传送 距离 总 
共 不 会 超过 2cm。 对 于 100GHz 的 计算 机 ,整个 传送 路 径 长 度 最 多 为 2mm。 而 在 一 台 
1THz(1000GHz) 的 计算 机 中 ,传送 距离 就 不 足 100pm 了 ,这 在 一 个 时 钟 周期 内 正好 让 信号 
从 一 端 到 另 一 端 并 返回 。 

让 计算 机 变 得 如 此 之 小 是 可 能 的 ,但 是 这 会 遇 到 另 一 个 基本 问题 : 散热 。 计 算 机 运行 
得 越 快 ,产生 的 热量 就 越 多 ,而 计算 机 越 小 就 越 难 散热 。 在 高 端 Pentium 系统 中 ,CPU 的 散 
热 器 已 经 比 CPU 自身 还 要 大 了 。 总 而 言 之 ,从 1MHz 到 1GHz 需要 的 是 更 好 的 芯片 制造 
工艺 ,而 从 1GHz 到 1THz 则 需要 完全 不 同 的 方法 。 

获得 更 高 速度 的 一 种 处 理 方式 是 大 规模 使 用 并 行 计算 机 。 这 些 机 器 有 许多 CPU ,每 一 
个 都 以 “通常 "的 速度 (在 一 个 给 定年 份 中 的 速度 ) 运 行 ,但 是 总 体 上 会 有 比 单个 CPU 强大 
得 多 的 计算 能 力 。 具 有 1000 个 CPU 的 系统 已 经 商业 化 了 。 在 未 来 十 年 中 ,可 能 会 建造 出 
具有 100 万 个 CPU 的 系统 。 当 然 为 了 获得 更 高 的 速度 ,还 有 其 他 潜在 的 处 理 方式 ,如 生物 
计算 机 ,但 在 本 章 中 ,我们 将 专注 于 有 多 个 普通 CPU 的 系统 。 


9.0 问题 导入 
多 处 理 机 使 系统 的 性 能 大 幅 提 升 ,那么 系统 中 的 多 个 处 理 机 应 如 何 组 织 呢 ? 当 一 个 复 


杂 的 、 多 线程 的 任务 需要 执行 时 ,系统 应 将 任务 分 配 到 哪些 处 理 器 上 执行 呢 ? 在 不 同 处 理 机 
上 的 各 个 线程 应 如 何 同步 呢 ? 


9.1 多 处 理 机 基本 概念 


要 讨论 多 核 环 境 下 的 操作 系统 所 做 的 调整 ,首先 需要 知道 多 核 环境 和 单 核 环 境 的 不 同 
之 处 。 首 先 来 看 一 下 多 核 的 一 些 基本 概念 。 在 x86 体系 结构 下 ,多 处 理 功能 芯片 经 过 了 对 


称 多 处 理 器 结构 (SMP Architecture)、 超 线程 结构 (Hyper Threading)、 多 核 结构 (Multi- 
core Architecture) 和 多 核 超 线程 结构 (Multi-core Hyper Threading Architecture) 的 4 个 演 
变 阶 段 。 下 面 分 别 予 以 介绍 。 


9.1.1 多 处 理 器 结构 


除了 提升 CPU 主 频 和 增加 一 、 二 级 缓存 容量 外 ,提升 计算 机 性 能 最 直截了当 的 办 法 
就 是 在 一 台 计 算 机 里 面 安装 多 个 CPU。 由 于 CPU 数量 增加 ,计算 机 同时 处 理 的 工作 量 
就 增加 ,自然 提升 了 系统 吞吐 量 和 改善 了 用 户 响 应 时 间 , 从 而 感觉 到 计算 机 的 性 能 得 到 
了 提升 。 

多 处 理 器 结构 说 简单 一 点 儿 就 是 在 一 条 总 线 上 挂 载 多 个 处 理 器 。 在 传统 的 体系 结构 
下 ,一 台 计算 机 里 面 只 有 一 个 CPU。 而 在 多 处 理 器 系统 里 ,一 台 计算 机 里 面 可 以 有 多 个 
CPU。 图 9-1 给 出 的 就 是 英特尔 公司 有 两 个 CPU 的 计算 机 体系 结构 。 
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图 9-1 双 处 理 器 结构 


在 多 CPU 的 情况 下 ,以 CPU 之 间 的 关系 不 同 又 可 以 分 为 对 称 和 非 对 称 多 处 理 器 结 
构 。 在 对 称 结构 下 ,多 个 CPU 角色 功能 平等 ,没有 主 从 之 分 ,这 种 多 CPU 结构 称 为 对 称 多 
处 理 器 结构 (Symmetric Multi-Processor Architecture,SMP) 。 在 非 对 称 多 处 理 器 结构 中 ， 
则 不 同 CPU 的 角色 地 位 不 同 ,有 主 从 CPU 之 分 。 这 种 多 CPU 结构 称 为 非 对 称 多 处 理 器 
结构 (Asymmetric Multi-Processor Architecture, AMP)。 由 于 SMP 结构 远 比 AMP 结构 
普遍 ,本 章 讨论 的 多 处 理 器 结构 将 只 对 SMP 进行 。 

当然 ,我 们 并 不 需要 限制 在 两 个 CPU 上 。 我 们 也 可 以 在 一 台 计 算 机 里 面 安装 多 于 两 
个 的 CPU。 图 9-2 给 出 的 就 是 一 台 有 着 4 个 CPU 的 计算 机 体系 结构 简化 


多 人 处理 机 系统 介绍 


攻 @ 但 


握 作 系统 原理 


cpuo | CPU1 cpu2| [ceus]| 


图 9-2 有 4 个 物理 CPU 的 对 称 多 处 理 器 结构 


9.1.2 超 线程 结构 


虽然 在 一 台 计 算 机 里 安装 多 个 CPU 提升 了 计算 机 的 性 能 ,但 是 付出 的 代价 是 高 昂 的 
成 本 和 巨大 的 功 耗 。 而 在 实际 中 ,基于 很 多 原因 ,CPU 的 执行 单元 并 没有 得 到 充分 使 用 。 
如 果 CPU 不 能 正常 读 取 数 据 ( 总 线 / 内 存 的 瓶颈 ), 其 执行 单元 利用 率 会 明显 下 降 。 另 外 就 
是 目前 大 多 数 执行 线程 缺乏 ILP(Instruction-Level Parallelism ,指令 级 并 发 , 即 多 条 指令 同 
时 执行 ) 支 持 。 这 些 都 造成 了 目前 CPU 的 性 能 没有 得 到 全 部 的 发 挥 。 因 此 ,英特尔 公司 提 
出 了 超 线程 技术 来 让 一 个 CPU 同时 执行 多 重 线程 ,从 而 提高 CPU 效率 和 用 户 满意 度 。 

超 线程 技术 是 在 一 个 CPU 上 同时 执行 多 个 程序 而 共享 这 个 CPU 内 的 资源 ,理论 上 要 
像 两 个 CPU 一 样 在 同一 时 间 执 行 两 个 线程 , 超 线程 技术 可 在 同一 时 间 里 让 应 用 程序 使 用 
芯片 的 不 同 部 分 。 而 为 了 支持 这 种 技术 ,需要 在 处 理 器 上 多 加 入 一 个 逻辑 处 理 单元 指针 
(Logical CPU Pointer) 。 因 此 新 一 代 的 P4 HT 的 模板 面积 比 以 往 的 P4 增 大 了 5%。 而 其 
余部 分 如 ALU、 浮 点 运算 单元 ,二 级 缓存 则 保持 不 变 。 

图 9-3 描述 的 是 超 线程 结构 。 图 中 的 CPU 并 不 是 物理 上 的 单个 CPU, 而 是 两 两 为 一 
个 独立 的 CPU, 即 图 中 只 有 4 个 物理 CPU ,而 每 个 CPU 又 因 超 线程 技术 分 解 为 两 个 逻辑 
CPU。 每 个 逻辑 CPU 可 以 执行 一 个 线程 序列 。 这 样 一 个 物理 CPU 可 以 同时 执行 两 个 
线程 。 


CPU0|CPU1| |cPU2[CPU3| |CPU4 
线程 0| 线 程 1 。 | 线程 0| 线 程 | [线程 0 


CPU5 CPU6|CPU7 
线程 1| ”| 线程 0| 线 程 1 
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IcH | 
图 9-3 4 个 物理 CPU.8 个 逻辑 CPU 的 超 线程 结构 


需要 注意 的 是 ,含有 超 线程 技术 的 CPU 需要 芯片 组 和 软件 的 支持 ,才能 比较 理想 地 发 
挥 该 项 技术 的 优势 。 操 作 系统 如 Windows XP、2003 以 及 Linux 2. 4. x 以 后 的 版 本 均 支 持 
超 线程 技术 。 

虽然 采用 超 线程 技术 能 同时 执行 两 个 线程 ,但 它 并 不 像 两 个 真正 的 CPU 那样 ,每 个 
CPU 都 具有 独立 的 资源 。 当 两 个 线程 都 同时 需要 某 一 资源 时 ,其 中 一 个 要 暂时 停止 ,并 让 
出 资源 ,直到 这 些 资源 闲置 后 才能 继续 。 因 此 超 线 程 的 性 能 并 不 等 于 两 个 CPU 的 性 能 。 


9.1.3 多 核 结构 


多 CPU 成 本 高 、 功 耗 大 ,而 超 线程 技术 又 不 等 于 两 个 CPU 的 性 能 ,而 且 时 常会 碰 到 两 
个 线程 需要 同一 资源 时 必须 停止 一 个 线程 的 现象 。 

那么 什么 办 法 可 以 同时 克服 多 CPU 和 超 线程 的 缺点 呢 ? 即 性 能 像 多 CPU 、 功 耗 像 超 
线程 的 结构 呢 ? 有 ,就 是 多 核 结构 。 

多 核 结 构 就 是 在 一 个 CPU 里 面 布置 两 个 执行 核 , 即 两 套 执行 单元 ,如 ALU、FPU 和 
L2 缓存 等 。 而 其 他 部 分 则 两 个 核 共 享 。 这样, 由 于 使 用 的 是 一 个 CPU, 其 功 耗 和 单 CPU 
一 样 。 由 于 布置 了 多 个 核 ,其 指令 级 并 行将 是 真正 的 并 行 ,而 不 是 超 线程 结构 的 半 并 行 。 

例如 ,英特尔 公司 的 奔腾 D 和 奔腾 EE( 见 


图 9-4) 即 是 分 别 面 向 主流 市 场 以 及 高 端 市 场 的 双 执行 核 执行 核 
核 芯 片 。 其 每 个 核 采 用 独立 式 缓存 设计 ,在 处 理 2MB L2 缓 存 2MB L2 缓 存 
器 内 部 两 个 核 之 间 是 互相 隔绝 的 ,通过 处 理 器 外 总 线 JIF 总 线 IF 
部 (主板 北桥 芯片 ) 的 仲裁 器 负责 两 个 核心 之 间 的 

任务 分 配 以 及 缓存 数据 的 同步 等 协调 工作 。 两 个 

核 共享 前 端 总 线 ,并 依靠 前 端 总 线 在 两 个 核 之 间 

传输 缓存 同步 数据 。 图 9-4 奔腾 D 和 奔腾 EE 双核 芯片 


当然 ,也 可 以 在 一 个 计算 机 里 面 放置 多 个 配置 
有 多 个 执行 核 的 CPU, 而 形成 更 多 的 核 。 例 如 ,如 果 用 4 个 双核 CPU 则 可 以 构建 如 图 9-5 所 


示 的 多 核 , 多 处 理 器 体系 结构 。 
Core0 | Corel Core0 | Corel Core0 | Corel 


Core0 | Corel 
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图 9-5 4 个 CPU .8 个 核 组 成 的 计算 机 体系 结构 
9.1.4 多 核 超 线程 结构 


在 多 核 情况 下 ,也 可 以 使 用 超 线 程 技 术 ,从 而 形成 多 核 超 线程 技术 。 即 每 个 物理 执行 核 
里 面 又 分 解 为 两 个 或 多 个 逻辑 执行 单元 ,如 图 9-6 所 示 。 


[Core0 [Corel Core0 | Corel Core0 | Corel Core0| Corel 
I [| | 
[mcn MEM 
ICH 


图 9-6 4 个 CPU.8 个 核 .16 个 逻辑 执行 单元 组 成 的 多 核 超 线程 结构 
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例如 ,英特尔 公司 的 奔腾 EE 多 核 芯片 就 支持 超 线程 技术 ,一 块 奔 腾 EE 芯片 在 打开 超 
线程 技术 之 后 会 被 操作 系统 识别 为 4 个 逻辑 处 理 器 。 


9.2 多 处 理 机 内 存 结构 


由 于 一 台 计 算 机 里 面 有 多 个 执行 核 ,而 每 个 执行 核 均 需要 对 内 存 进行 访问 。 那 么 这 种 
访问 在 多 个 核 之 间 是 如 何 协调 的 呢 ? 或 者 说 内 存在 多 个 核 之 间 是 如 何 分 配 的 呢 ? 


9.2.1 UMA 结构 


最 简单 的 内 存 共享 方式 就 是 将 内 存 作 为 与 执行 核 独 立 的 单元 构建 在 核 之 外 ,所 有 的 核 
通过 同一 总 线 对 内 存 进行 访问 。 由 于 每 个 核 使 用 相同 的 方式 访问 内 存 ,其 到 内 存 的 延迟 也 
相同 ,这 种 访问 模式 称 为 均匀 内 存 访问 (Uniform Memory Access,UMA)。 在 这 种 模式 下 ， 
最 重要 的 是 所 有 核 的 地 位 在 内 存 面前 平等 。 其 优点 是 设计 简单 ,实现 容易 。 缺 点 是 大 锅 饭 ， 
难以 针对 个 体 的 程序 进行 访问 优化 ,以 及 扩展 困难 。 随 着 执行 核 数量 的 增加 ,对 共享 内 存 的 
竞争 将 变 得 白热化 ,从 而 造成 系统 效率 急剧 下 降 。 

当前 的 对 称 多 处 理 器 共享 存储 系统 基本 上 采用 此 种 模式 。 这 种 模式 只 能 在 处 理 器 个 数 
或 执行 核 数 量 较 少时 方 可 使 用 。 


9.2.2 NUMA 结构 


如 果 想 构建 CPU 数量 很 多 的 多 处 理 器 系统 ,或 者 欲 构建 执行 核 多 于 4 个 以 上 的 多 核 系 
统 , 则 UMA 结构 因 内 存 共享 瓶颈 而 不 能 胜任 。 在 这 种 情况 下 ,一 种 自然 的 选择 是 使 用 多 个 
分 开 的 独立 共享 内 存 。 每 个 执行 核 或 CPU 到 达 不 同 共享 内 存 的 距离 不 同 , 访 问 延迟 也 不 
一 样 。 这 种 访问 延迟 不 一 致 的 内 存 共享 模式 称 为 非 均匀 内 存 访问 (Non-Uniform Memory 
Access, NUMA)。 在 这 种 模式 下 ,最 重要 的 特点 是 执行 核 在 不 同 的 内 存单 元 面前 地 位 并 不 
平等 ; 到 近 的 内 存 具 有 优势 地 位 ,到 远 的 内 存 则 处 于 劣势 。 

在 NUMA 下 ,原则 上 应 该 将 程序 调度 到 离 本 地 内 存 ( 程 序 存放 的 内 存单 元 ) 近 的 执行 
核 上 ,以 提升 程序 的 内 存 访 问 效率 ,从 而 提高 程序 的 执行 效率 。 

NUMA 结构 的 优点 是 灵活 性 高 .扩展 容易 。 在 执行 核 的 数量 增加 的 时 候 ,其 访问 内 存 
的 效率 可 以 保持 不 下 降 。 不 过 ,这 种 不 下 降 的 前 提 是 优良 的 调度 策略 , 即 在 调度 时 能 够 将 程 
序 就 近 执行 。 和 否则 ,有 可 能 因 内 存 访问 距离 远 而 造成 效率 下 降 。 因 此 ,NUMA 对 调度 的 要 
求 很 高 。 但 因为 扩展 容易 ,所 以 NUMA 得 到 了 非常 广泛 的 应 用 。 

如 图 9-7 所 示 , 是 NUMA 的 原理 图 ,具有 8 核心 的 处 理 器 可 分 为 两 个 NUMA 结 点 。 
英特尔 公司 以 NUMA 方式 已 研发 出 80 个 核 的 芯片 ,其 尺寸 并 不 比 一 个 指甲 盖 大 多 少 ,该 
80 核心 处 理 器 具有 万 亿 次 浮 点 运算 能 力 , 它 也 是 世界 首 个 具有 万 亿 次 浮 点 运算 能 力 的 可 编 
程 计算 器 。 


9.2.3 COMA 结构 


前 面 说 过 ,NUMA 具有 灵活 、 易 扩展 的 优点 ,但 对 调度 的 要 求 高 。 如 果 不 能 或 难以 将 
程序 调度 到 就 近 的 执行 核 上 , 那 又 怎么 办 呢 ? 答案 是 缓存 。 即 在 每 个 执行 核 里 面 配置 缓存 ， 


Direct 
Access 


图 9-7 NUMA 原理 图 


其 执行 需要 的 数据 均 缓 存在 该 缓存 里 面 。 所 有 访问 由 缓存 得 到 满足 。 这 样 ,不 论 数据 原来 
是 处 于 哪个 内 存单 元 ,其 对 效率 的 影响 均 将 不 复 存 在 。 

这 种 完全 由 缓存 满足 数据 访问 的 模式 称 为 全 缓存 内 存 访问 (Cache Only Memory 
Access,COMA)。 在 这 种 模式 下 ,每 个 执行 核 配备 的 缓存 共同 组 成 全 局 地 址 空间 。 


9.2.4 NORMA 结构 


如 果 内 存单 元 为 每 个 执行 核 所 私有 , 且 每 个 执行 核 只 能 访问 自己 的 私有 内 存 , 对 其 他 内 
存单 元 的 访问 通过 消息 传递 进行 , 则 就 是 非 远 程 内 存 访问 模式 (Non-Remote Memory 
Access,NORMA)。 这 种 模式 的 优点 是 设计 比 NUMA 还 要 简单 ,但 执行 核 之 间 的 通信 成 
本 高 晶 。 这 已 经 有 一 点 儿 像 网 络 了 。 因 为 效率 问题 ,这 种 模式 在 多 核 体系 结构 下 使 用 
甚 少 。 


9.3 多 处 理 机 操作 系统 类 型 


让 我 们 从 对 多 处 理 机 硬件 的 讨论 转 到 多 处 理 机 软件 ,特别 是 多 处 理 机 操作 系统 上 来 。 
这 里 有 各 种 可 能 的 方法 。 接 下 来 将 讨论 其 中 的 三 种 。 这 些 方 法 除了 适用 于 多 核 系统 之 外 ， 
同样 适用 于 包含 多 个 分 离 CPU 的 系统 。 

1. 每 个 CPU 有 自己 的 操作 系统 

组 织 一 个 多 处 理 机 操作 系统 可 能 的 最 简单 的 方法 是 ,静态 地 把 存储 器 划分 成 和 CPU 
一 样 多 的 各 个 部 分 ,为 每 个 CPU 提供 私有 存储 器 以 及 操作 系统 的 各 自私 有 副本 。 实 际 上 ? 
个 CPU 以 nn 个 独立 计算 机 的 形式 运行 。 这 样 做 一 个 明显 的 优点 是 ,允许 所 有 的 CPU 共享 
操作 系统 的 代码 ,而 且 只 需要 提供 数据 的 私有 副本 ,如 图 9-8 所 示 。 


CPUI CPU2 CPU3 CPU4 存储 器 UO 
有 OS 的 | | 有 os 的 | | 有 os 的 | | 有 os | 一 
OS 的 j OS 的 7 OS 的 OS 的 3 
私有 副本 | | 私有 副本 | | 私有 副本 | | 私有 副本 汪汪 
O: 
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图 9-8 在 4 个 CPU 中 划分 多 处 理 机 存储 器 ,但 共享 一 个 OS 代码 的 副本 
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这 一 机 制 比 有 个 分 离 的 计算 机 要 好 ,因为 它 允 许 所 有 的 机 器 共享 一 套 磁盘 及 其 他 的 
1/O 设备 , 它 还 允许 灵活 地 共享 存储 器 。 例 如 ,即便 使 用 静态 内 存 分 配 ,一 个 CPU 也 可 以 获 
得 极 大 的 一 块 内 存 , 从 而 高 效 地 执行 代码 。 另 外 ,由 于 生产 者 能 够 直接 把 数据 写 入 存储 器 ， 
从 而 使 得 消费 者 从 生产 者 写 入 的 位 置 取 出 数据 ,因此 进程 之 间 可 以 高 效 地 通信 。 况 且 , 从 操 
作 系 统 的 角度 看 ,CPU 都 有 自己 的 操作 系统 非常 自然 。 

值得 提 及 该 设计 看 来 不 明显 的 4 个 方面 。 

第 一 ,在 一 个 进程 进行 系统 调用 时 ,该 系统 调用 是 在 本 机 的 CPU 上 被 捕获 并 处 理 的 ， 
使 用 操作 系统 表 中 的 数据 结构 。 

第 二 ,因为 每 个 操作 系统 都 有 自己 的 表 , 那 么 它 也 有 自己 的 进程 集合 ,通过 自身 调度 这 
些 进程 。 这 里 没有 进程 共享 。 如 果 一 个 用 户 登 录 到 CPU1, 那 么 他 的 所 有 进程 都 在 CPU1 
上 运行 。 因 此 ,在 CPU1 有 负载 运行 而 CPU2 空 载 的 情形 是 会 发 生 的 。 

第 三 ,没有 页 面 共 享 。 会 出 现 如 下 的 情形 : 在 CPU2 不 断 地 进行 页 面 调度 时 CPU1 却 
有 多 余 的 页 面 。 由 于 内 存 分 配 是 固定 的 ,所 以 CPU2 无 法 向 CPU1 借用 页 面 。 

第 四 ,也 是 最 坏 的 情形 ,如 果 操作 系统 维护 近期 使 用 过 的 磁盘 块 的 缓冲 区 高 速 缓存 ,每 
个 操作 系统 都 独自 进行 这 种 维护 工作 ,因此 ,可 能 出 现 某 一 修改 过 的 磁盘 块 同时 存在 于 多 个 
缓冲 区 高 速 缓存 的 情况 ,这 将 会 导致 不 一 致 性 的 结果 。 避 免 这 一 问题 的 唯一 途径 是 ,取消 组 
冲 区 高 速 缓存 。 这 样 做 并 不 难 ,但 是 会 显著 降低 性 能 。 

由 于 这 些 原因 ,上 述 模型 已 很 少 使 用 ,尽管 在 早期 的 多 处 理 机 中 它 一 度 被 采用 , 那 时 的 
目标 是 把 已 有 的 操作 系统 尽 可 能 快 地 移植 到 新 的 多 机 理 机 上 。 

2. 主 从 多 处 理 机 

图 9-9 中 给 出 的 是 第 二 种 模型 。 在 这 种 模型 中 ,操作 系统 的 一 个 副本 及 其 数据 表 都 在 
CPU1 上 ,而 不 是 在 其 他 所 有 CPU 上 。 为 了 在 该 CPU1 上 进行 处 理 , 所 有 的 系统 调用 都 重 
定向 到 CPU1 上 。 如 果 有 剩余 的 CPU 时 间 , 还 可 以 在 CPU1 上 运行 用 户 进程 。 这 种 模型 
称 为 主 从 模型 (Master-Slave) ,因为 CPU1 是 主 CPU, 而 其 他 的 都 是 从 属 CPU。 


CPUI CPU2 CPU3 CPU4 存储 器 UO 
主机 运行 | | 从 机 运行 | | 从 机 运行 | | 从 机 运行 用 户 进程 
操作 系统 | | 用 户 过 程 | | 用 户 过 程 | | 用 户 过 程 

OS 代码 
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图 9-9 主 从 多 处 理 机 模型 


主 从 模型 解决 了 在 第 一 种 模型 中 的 多 数 问题 。 有 单一 的 数据 结构 (如 一 个 链表 或 者 一 
组 优先 级 链表 ) 用 来 记录 就 绪 进 程 。 当 某 个 CPU 空闲 下 来 时 , 它 向 CPU1 上 的 操作 系统 请 
求 一 个 进程 运行 ,并 被 分 配 一 个 进程 。 这 样 ,就 不 会 出 现 一 个 CPU 空闲 而 另 一 个 过 载 的 情 
形 。 类 似 地 ,可 在 所 有 的 进程 中 动态 地 分 配 页 面 ,而 且 只 有 一 个 缓冲 区 高 速 缓 存 ,所 以 绝 不 
会 出 现 不 一 致 的 情形 。 

这 个 模型 的 问题 是 ,如 果 有 很 多 的 CPU, 主 CPU 会 变 成 一 个 瓶颈 。 毕 竟 , 它 要 处 理 来 
自 所 有 CPU 的 系统 调用 。 如 果 全 部 时 间 的 10% 用 来 处 理 系统 调用 ,那么 10 个 CPU 就 会 


使 主 CPU 饱和 ,而 20 个 CPU 就 会 使 主 CPU 彻底 过 载 。 可 见 , 这 个 模型 虽然 简单 ,而 且 对 
小 型 多 处 理 机 是 可 行 的 ,但 不 能 用 于 大 型 多 处 理 机 。 

3. 对 称 多 处 理 机 

第 三 种 模型 , 即 对 称 多 处 理 机 (Symmetric MultiProcessor, SMP) ,消除 了 上 述 的 不 对 称 
性 。 在 存储 器 中 有 操作 系统 的 一 个 副本 ,但 任何 CPU 都 可 以 运行 它 。 在 有 系统 调用 时 , 进 
行 系统 调用 的 CPU 陷入 内 核 并 处 理 系 统 调用 。 图 9-10 是 对 SMP 模式 的 说 明 。 


CPU1 CPU2 CPU3 CPU4 存储 器 VO 


运行 用 户 | | 运行 用 户 | | 运行 用 户 | | 运行 用 户 
进程 和 共 | | 进程 和 共 | | 进程 和 共 | | 进程 和 共 
享 操作 系 绝 | 享 操作 系 绝 | 享 操作 系统 | | 享 操作 系统 


总 线 


图 9-10 SMP 多 处 理 机 模型 


Os 代码 | 鲁 


这 个 模型 动态 地 平衡 进程 和 存储 器 ,因为 它 只 有 一 套 操作 系统 数据 表 。 它 还 消除 了 主 
CPU 的 瓶颈 ,因为 不 存在 主 CPU; 但 是 这 个 模型 也 带 来 了 自身 的 问题 。 特 别 是 , 当 两 个 或 
更 多 的 CPU 同时 运行 操作 系统 代码 时 ,就 会 出 现 灾难 。 想 象 有 两 个 CPU 同时 选择 相同 的 
进程 运行 或 请 求 同 一 个 空闲 存储 器 页 面 。 处 理 这 些 问 题 的 最 简单 方法 是 在 操作 系统 中 使 用 
互 斥 信号 量 ( 锁 ) ,使 整个 系统 成 为 一 个 大 临界 区 。 当 一 个 CPU 要 运行 操作 系统 时 , 它 必 须 
首先 获得 互 斥 信号 量 。 如 果 互 斥 信号 量 被 锁 住 , 就 得 等 待 。 按 照 这 种 方式 ,任何 CPU 都 可 
以 运行 操作 系统 ,但 在 任 一 时 刻 只 有 一 个 CPU 可 以 运行 操作 系统 。 

这 个 模型 是 可 以 工作 的 ,但 是 它 几 乎 同 主 从 模式 一 样 的 糟糕 。 同 样 假设 ,如 果 所 有 时 间 
的 10% 花 费 在 操作 系统 内 部 。 那 么 在 有 20 个 CPU 时 ,会 出 现 等 待 进入 的 CPU 长 队 。 幸 
运 的 是 ,比较 容易 进行 改进 。 操 作 系 统 中 的 很 多 部 分 是 彼此 独立 的 。 例 如 ,在 一 个 CPU 运 
行 调度 程序 时 , 另 一 个 CPU 则 处 理 文件 系统 的 调用 ,而 第 三 个 在 处 理 一 个 缺 页 异常 ,这 种 
运行 方式 是 没有 问题 的 。 

这 一 事实 使 得 把 操作 系统 分 割 成 互 不 影响 的 临界 区 。 每 个 临界 区 由 其 互 斥 信号 量 保 
护 ,所 以 一 次 只 有 一 个 CPU 可 执行 它 。 采 用 这 种 方式 ,可 以 实现 更 多 的 并 行 操作 。 而 某 些 
表格 ,如 进程 表 , 可 能 恰巧 被 多 个 临界 区 使 用 。 例 如 ,在 调度 时 需要 进程 表 , 在 系统 fork 调 
用 和 信号 处 理 时 也 都 需要 进程 表 。 多 临界 区 使 用 的 每 个 表格 ,都 需要 有 各 自 的 互 斥 信号 量 。 
通过 这 种 方式 ,可 以 做 到 每 个 临界 区 在 任 一 时 刻 只 被 一 个 CPU 执行 ,而 且 在 任 一 时 刻 每 个 
临界 表 也 只 被 一 个 CPU 访问 。 

大 多 数 的 现代 多 处 理 机 都 采用 这 种 安排 。 为 这 类 机 器 编写 操作 系统 的 困难 ,不 在 于 其 
实际 的 代码 与 普通 的 操作 系统 有 多 大 的 不 同 , 而 在 于 如 何 将 其 划分 为 可 以 由 不 同 的 CPU 
并 行 执行 的 临界 区 而 互 不 干扰 ,即使 以 细小 的 、 间 接 的 方式 。 另 外 ,对 于 被 两 个 或 多 个 临界 
区 使 用 的 表 必 须 通过 互 斥 信号 量 分 别 加 以 保护 ,而 且 使 用 这 些 表 的 代码 必须 正确 地 运用 互 
斥 信 号 量 。 

更 进一步 ,必须 格外 小 心地 避免 死 锁 。 如 果 两 个 临界 区 都 需要 表 A 和 表 B, 其 中 一 个 
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首先 申请 A, 另 一 个 首先 申请 B, 那 么 迟早 会 发 生死 锁 , 而 且 没 有 人 知道 为 什么 会 发 生死 锁 。 
理论 上 ,所 有 的 表 可 以 被 赋予 整数 值 ,而 且 所 有 的 临界 区 都 应 该 以 升序 的 方式 获得 表 。 这 一 
策略 避免 了 死 锁 , 但 是 需要 程序 员 非 常 仔细 地 考虑 每 个 临界 区 需要 哪个 表 , 以便 按照 正确 的 
次 序 安排 请 求 。 

由 于 代码 是 随 着 时 间 演 化 的 ,所 以 也 许 有 个 临界 区 需要 一 张 过 去 不 需要 的 新 表 。 如 果 
程序 员 是 新 接手 工作 的 ,他 不 了 解 系统 的 整个 逻辑 ,那么 可 能 只 是 在 他 需要 的 时 候 获 得 表 ， 
并 且 在 不 需要 时 释放 掉 。 尽 管 这 看 起 来 是 合理 的 ,但 是 可 能 会 导致 死 锁 , 即 用 户 会 觉察 到 系 
统 被 凝固 住 了 。 要 做 正确 并 不 容易 ,而 且 要 在 程序 员 不 断 更 换 的 数 年 时 间 之 内 始终 保持 正 
确 性 太 困 难 了 。 


9.4 多 处 理 器 之 间 的 通信 


既然 一 个 系统 里 有 多 个 CPU ,这 些 CPU 之 间 总 需要 进行 某 种 通信 ,以 进行 任务 协调 。 
而 这 种 协调 既 可 能 是 CPU 本 身 需 要 ,也 可 能 是 运行 在 它们 上 面 的 进程 和 线程 之 间 需 要 。 
那么 CPU 之 间 的 通信 方式 是 什么 呢 ? 

我 们 讲 进程 时 讨论 过 进程 间 的 通信 : 信和 号、 信号 量 、 消 息 队 列 、 管 道 \ 共 享 内 存 等 。 这 些 
机 制 能 否 用 来 实现 多 CPU 之 间 的 通信 呢 ? 在 多 CPU 之 间 通 信 , 自 然 也 可 以 发 送信 号 。 不 
过 这 个 信号 不 是 内 存 的 一 个 对 象 ,因为 这 样 ,就 无 法 及 时 引起 另外 一 个 CPU 的 注意 。 而 要 
引起 其 注意 ,需要 发 送 的 是 中 断 。 

用 来 协调 这 些 CPU 之 间 中 断 的 机 制 就 是 高 级 可 编程 中 断 控制 器 (APIC)。 这 是 实现 
SMP 功能 必 不 可 少 的 , 且 是 英特尔 多 处 理 器 规范 的 核心 。 在 此 种 规范 下 ,每 个 CPU 内 部 必 
须 内 置 APIC 单元 (成 为 那个 CPU 的 本 地 APIC)。CPU 通过 彼此 发 送 中 断 (IPI, 即 处 理 器 
间 中 断 ) 来 完成 它们 之 间 的 通信 。 通 过 给 中 断 附加 动作 ,不 同 的 CPU 可 以 在 某 种 程度 上 彼 
此 进行 控制 。 

除了 每 个 CPU 自己 本 地 的 APIC 外 ,所 有 CPU 通常 还 共享 一 个 IJO APIC 来 处 理由 
1/O 设备 引起 的 中 断 , 这 个 IJO APIC 是 安装 在 主板 上 的 。 图 9-11 描述 的 是 英特尔 公司 的 
Xeon 多 处 理 器 结构 下 的 本 地 APIC 和 I/O APIC 的 结构 示意 图 。 


CPU1 CPU2 CPU3 CPU4 
本 地 APIC 本 地 APIC 本 地 APIC 本 地 APIC 
中 断 消息 人 和 IP[ 中 断 消息 下 全 TPL 中 i 消息 是 IPL 中 是 消息 人 人 
中 电 消 息 人 处 理 器 系统 总 线 
桥接 器 
PCI 总 线 


IO APIC 碑 一 一 外 部 中 断 


系统 芯片 组 
图 9-11 Xeon 里 的 本 地 APIC 和 IO APIC 


在 目前 的 建 置 中 ,系统 的 每 一 个 部 分 都 是 经 由 APIC 总 线 连 接 的 。“ 本 机 APIC” 为 系统 
的 一 部 分 ,负责 传递 中 断 至 指定 的 处 理 器 。 例 如 , 当 一 台 机 器 上 有 4 个 处 理 器 , 则 它 必 须 相 
应 地 要 有 4 个 本 机 APIC。 自 1994 年 的 Pentium P54c 开始 ,英特尔 已 经 将 本 机 APIC 建 置 
在 其 处 理 器 中 。 实 际 建 置 了 英特尔 处 理 器 的 计算 机 就 已 经 包含 APIC 系统 的 部 分 。 

系统 中 另外 一 个 重要 的 部 分 为 IO APIC。 系 统 中 最 多 可 拥有 8 个 IO APIC, 它 们 会 
收集 来 自 1/O 装置 的 中 断 信 号 且 在 当 那 些 装置 需要 中 断 时 传送 信息 至 本 机 APIC。 每 个 
1/O APIC 有 一 个 专 有 的 中 断 输 入 (或 IRQ) 号 码 。 英 特 尔 过 去 与 目前 的 IO APIC 通常 有 
24 个 输入 ,其 他 的 可 能 有 多 达 64 个 。 而 且 有 些 机 器 拥有 数 个 II/O APIC ,每 一 个 都 有 自己 
的 输入 号 码 , 加 起 来 一 台 机 器 上 会 有 上 百 个 IRQ 可 供 中 断 使 用 。 

除了 处 理 处 理 器 间 及 输入 输出 的 中 断 外 ,APIC 也 负责 处 理 本 地 中 断 源 发 出 的 中 断 ,如 
本 地 连接 的 MO 设备 .时 序 中 断 、 性 能 监视 计数 器 中 断 ,高 温 中 断 .内 部 错误 中 断 等 。 


9.5 多 处 理 机 同步 


在 多 处 理 机 中 CPU 经 常 需要 同步 。 前 面 看 到 了 内 核 临 界 区 和 表 被 互 斥 信号 量 保护 的 
情形 。 现 在 让 我 们 仔细 看 看 在 多 处 理 机 中 这 种 同步 是 如 何 工作 的 。 

开始 讨论 之 前 ,还 需要 引入 同步 原 语 。 如 果 一 个 进程 在 单 处 理 机 ( 仅 含 一 个 CPU) 中 需 
要 访问 一 些 内 核 临 界 表 的 系统 调用 ,那么 内 核 代码 在 接触 该 表 之 前 可 以 先 禁止 中 断 。 然 后 
它 继续 工作 ,在 相关 工作 完成 之 前 ,不 会 有 任何 其 他 的 进程 溜 进来 访问 该 表 。 大 多 处 理 机 
中 ,禁止 中 断 的 操作 只 影响 到 完成 禁止 中 断 操作 的 这 个 CPU ,其 他 的 CPU 继续 运行 并 且 可 
以 访问 临界 表 。 因 此 ,必须 采用 一 种 合适 的 互 斥 信号 量 协议 ,而 且 所 有 的 CPU 都 遵守 该 协 
议 以 保证 互 斥 工作 的 进行 。 

任何 实用 的 互 斥 信号 量 协议 的 核心 都 是 一 条 特殊 指令 ,该 指令 允许 检测 一 个 存储 器 字 
并 以 一 种 不 可 见 的 操作 设置 。 我 们 看 看 在 图 3-1 中 使 用 的 exchange 指令 (XCHG) 是 如 何 
实现 临界 区 的 。 正 如 先前 讨论 的 ,这 条 指令 做 的 是 , 读 出 一 个 存储 单元 的 内 容 并 把 它 存 储 在 
一 个 寄存 器 中 ,同时 ,对 该 存储 单元 写 入 一 个 1( 或 某 个 非 零 值 )。 当 然 , 这 需要 两 个 总 线 周 
期 来 完成 存储 器 的 读 写 。 在 单 处 理 机 中 ,只 要 该 指令 不 被 中 途中 断 ,XCHG 指令 就 始终 正 
常 工 作 。 

现在 考虑 在 一 个 多 处 理 机 中 发 生 的 情况 ,在 图 9-12 中 我 们 看 到 了 最 坏 情 况 的 时 序 ,其 
中 存储 器 字 2000 ,被 用 作 一 个 初始 化 为 0 的 锁 。 第 1 步 ,CPU1 读 出 该 字 得 到 一 个 0。 第 2 
步 , 在 CPU1 有 机 会 把 该 字 写 为 1 之 前 ,CPU2 进入 ,并 且 它 读 出 该 字 为 0。 第 3 步 ,CPUI1 
把 1 写 入 该 字 。 第 4 步 ,CPU2 把 1 写 入 该 字 。 两 个 CPU 都 由 XCHG 指令 得 到 0, 所 以 两 
者 都 对 临界 区 进行 访问 ,并 且 互 斥 失 败 。 


CPU1 


2000 


| 0 
| 1. CPU1 读 一 个 0 


2. CPU2 读 一 个 0 


3. CPU1 写 一 个 1 4.CPU2 写 一 个 1 


图 9-12 如 果 不 能 锁 住 总 线 , 比 较 和 交换 指令 会 失效 
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为 了 阻止 这 种 情况 的 发 生 ,XCHG 指令 必须 首先 锁 住 总 线 , 阻 止 其 他 的 CPU 访问 它 ， 
然后 进行 存储 器 的 读 写 访问 ,再 解锁 总 线 。 对 总 线 加 锁 的 典型 做 法 是 , 先 使 用 通常 的 总 线 协 
议 请 求 总 线 , 并 申明 (设置 一 个 逻辑 1) 已 拥有 某 些 特定 的 总 线 线路 ,直到 两 个 周期 全 部 完 
成 。 只 要 始终 保持 拥有 这 一 特定 的 总 线 线路 ,那么 其 他 CPU 就 不 会 得 到 总 线 的 访问 权 。 
这 个 指令 只 有 在 拥有 必要 的 线路 和 使 用 它们 的 (硬件 ) 协 议 上 才能 实现 。 现 代 总 线 有 这 些 功 
能 ,但 是 早期 的 一 些 总 线 不 具备 ,它们 不 能 正确 地 实现 XCHG 指令 。 这 就 是 Peterson 协议 
(完全 用 软件 实现 同步 ) 会 产生 的 原因 。 

如 果 正 确 地 实现 和 使 用 XCHG 指令 , 它 能 够 保证 互 斥 机 制 正常 工作 。 但 是 这 种 互 斥 方 
法 使 用 了 自 旋 锁 ,因为 请 求 的 CPU 只 在 原 地 尽 可 能 快 地 对 锁 进 行 循环 测试 。 这 样 做 不 仅 
完全 浪费 了 提出 请 求 的 各 个 CPU 的 时 间 , 而 且 还 给 总 线 或 存储 器 增加 了 大 量 的 负载 ,严重 
地 降低 了 所 有 其 他 CPU 从 事 正常 工作 的 速度 。 

乍 一 看 ,高 速 缓存 的 实现 也 许 能 够 消除 总 线 竞 争 的 问题 ,但 事实 并 非 如 此 。 理 论 上 ,只 
要 提出 请 求 的 CPU 已 经 读 取 了 锁 字 , 它 就 可 在 其 高 速 缓存 中 得 到 一 个 副本 。 只 要 没有 其 
他 CPU 试图 使 用 该 锁 , 提 出 请 求 的 CPU 就 能 够 用 完 其 高 速 缓存 。 当 拥有 锁 的 CPU 写 入 
一 个 1 到 高 速 缓存 并 释放 它 时 ,高 速 缓存 协议 会 自动 地 将 它 在 远程 高 速 缓存 中 的 所 有 副本 
失效 ,要 求 再 次 读 取 正确 的 值 。 

问题 是 ,高 速 缓存 操作 是 在 32 字 节 或 64 字 节 的 块 中 进行 的 ,通常 ,拥有 锁 的 CPU 也 需 
要 这 个 锁 周围 的 字 。 由 于 XCHG 指令 是 一 个 写 指令 (因为 它 修改 了 锁 ) ,所 以 它 需要 互 斥 地 
访问 含有 锁 的 高 速 缓存 块 。 这 样 , 每 一 个 XCHG 都 使 锁 持 有 者 的 高 速 缓存 中 的 块 失效 ,并 
且 为 请 求 的 CPU 取 一 个 私有 的 、 唯 一 的 副本 。 只 要 锁 拥 有 者 访问 到 该 锁 的 邻接 字 , 该 高 速 
缓存 块 就 被 送 进 其 机 器 。 这 样 一 来 ,整个 包含 锁 的 高 速 缓存 块 就 会 不 断 地 在 锁 的 拥有 者 入 
锁 的 请 求 者 之 间 来 回 穿梭 ,导致 了 比 单个 读 取 一 个 锁 字 更 大 的 总 线 流 量 。 

如 果 能 消除 在 请 求 一 侧 的 所 有 由 XCHG 引起 的 写 操作 ,就 可 以 明显 地 减少 这 种 开销 。 
使 提出 请 求 的 CPU 首先 进行 一 个 纯 读 操作 来 观察 锁 是 否 空闲 ,就 可 以 实现 这 个 目标 。 只 
有 在 锁 看 来 是 空闲 时 ,XCHG 才 真正 去 获取 它 。 这 种 小 小 变化 的 结果 是 ,大 多 数 的 行为 变 
成 读 而 不 是 写 。 如 果 拥 有 锁 的 CPU 只 是 在 同一 个 高 速 缓存 块 中 读 取 各 种 变量 ,那么 它们 
每 个 都 可 以 以 共享 只 读 方式 拥有 一 个 高 速 缓存 块 的 副本 ,这 就 消除 了 所 有 的 高 速 缓存 块 传 
送 。 当 锁 最 终 被 释放 时 , 锁 的 所 有 者 进行 写 操作 ,这 需要 排他 访问 ,也 就 使 远程 高 速 缓存 中 
的 所 有 其 他 副本 失效 。 在 提出 请 求 的 CPU 的 下 一 个 读 请 求 中 ,高 速 缓存 块 会 被 重新 装载 。 
注意 ,如 果 两 个 或 更 多 的 CPU 竞争 同一 个 锁 , 那 么 有 可 能 出 现 这 样 的 情况 ,两 者 同时 看 到 
锁 是 空闲 的 ,于 是 同时 用 XCHG 指令 去 获得 它 。 只 有 其 中 一 个 会 成 功 ,所 以 这 里 没有 竞争 
条 件 , 因 为 真正 的 获取 是 由 XCHG 指令 进行 的 ,而 且 这 条 指令 是 原子 性 的 ,即使 看 到 了 锁 空 
闲 , 然 后 立即 用 XCHG 指令 试图 获得 它 , 也 不 能 保证 真正 得 到 它 。 其 他 CPU 可 能 会 取胜 ， 
不 过 对 于 该 算法 的 正确 性 来 说 , 谁 得 到 了 锁 并 不 重要 。 纯 读 出 操作 的 成 功 只 是 意味 着 这 可 
能 是 一 个 获得 锁 的 好 时 机 ,但 并 不 能 确保 能 成 功 地 得 到 锁 。 

另 一 个 减少 总 线 流量 的 方式 是 使 用 著名 的 以 太 网 二 进 制 指数 补偿 算法 。 不 是 采用 连续 
轮 询 , 而 是 把 一 个 延迟 循环 插入 轮 询 之 间 。 初 始 的 延迟 是 一 条 指令 。 如 果 锁 仍然 忙 ,延迟 被 
加 倍 成 为 两 条 指令 ,然后 ,4 条 指令 ,如 此 这 样 进行 ,直到 某 个 最 大 值 。 当 锁 释 放 时 , 较 低 的 
最 大 值 会 产生 快速 的 响应 。 但 是 会 浪费 较 多 的 总 线 周 期 在 高 速 缓存 的 颠 艇 上 。 而 较 高 的 最 


大 值 可 减少 高 速 缓存 的 颠 篮 ,但 是 其 代价 是 不 会 注意 到 锁 如 此 迅速 地 成 为 空闲 。 二 进 制 指 
数 补偿 算法 无 论 在 有 或 无 XCHG 指令 前 的 纯 读 的 情况 下 都 适用 。 

一 个 更 好 的 思想 是 ,让 每 个 打算 获得 互 斥 信号 量 的 CPU 都 拥有 各 自用 于 测试 的 私有 
锁 变 量 , 如 图 9-13 所 示 。 有 关 的 变量 应 该 存放 在 未 使 用 的 高 速 缓存 块 中 以 避免 冲突 。 对 这 
种 算法 的 描述 如 下 : 给 一 个 未 能 获得 锁 的 CPU 分 配 一 个 锁 变量 并 把 它 附 在 等 待 该 锁 的 
CPU 链表 的 末端 。 在 当前 锁 的 持 有 者 退出 临界 区 时 , 它 释 放 链 表 中 的 首 个 CPU 正在 测试 
的 私有 锁 ( 在 自己 的 高 速 缓 存 中 )。 然 后 该 CPU 进入 临界 区 。 操 作 完 成 之 后 ,该 CPU 释放 
锁 。 其 后 继 者 接着 使 用 ,以 此 类 推 ,尽管 这 个 协议 有 些 复杂 (为 了 避免 两 个 CPU 同时 把 它 
们 自己 加 在 链表 的 末端 ) ,但 它 能 够 有 效 工 作 , 而 且 消 除了 饥饿 问题 。 

当 CPU1 在 实际 锁 上 完成 时 ， 它 
释放 该 锁 ， 同 时 也 释放 CPU2 正 
在 其 上 轮转 的 私有 锁 


CPU1 持 有 实际 锁 


加 
CPU4 在 这 个 锁 
(私有 ) 上 轮转 


CPU2 在 这 个 锁 
(私有 ) 上 轮转 


CPU3 在 这 个 锁 
(私有 ) 上 轮转 


图 9-13 ”使 用 多 个 锁 以 防止 高 速 缓存 舌 艇 


到 目前 为 止 , 不 论 是 连续 轮 询 方式 , 间 鞭 轮 询 方式 ,还 是 把 自己 附 在 进行 等 候 CPU 链 
表 中 的 方式 ,我 们 都 假定 需要 加 锁 的 互 斥 信号 量 的 CPU 只 是 保持 等 待 。 有 时 对 于 提出 请 
求 的 CPU 而 言 ,只 有 等 待 , 不 存在 其 他 替代 的 办 法 。 例 如 ,假设 一 些 CPU 是 空闲 的 ,需要 
访问 共享 的 就 绪 链 表 以 便 选 择 一 个 进程 运行 。 如 果 就 绪 链表 被 锁 住 了 ,那么 CPU 就 不 能 
够 只 是 决定 暂停 其 正在 进行 的 工作 ,而 去 运行 另 一 个 进程 ,因为 这 样 做 需要 访问 就 绪 链表 。 
CPU 必须 保持 等 待 直到 能 够 访问 该 就 绪 链表 。 

然而 ,在 另外 一 些 情形 中 , 却 存 在 着 别 的 选择 。 例 如 ,如 果 在 一 个 CPU 中 的 某 些 线程 
需要 访问 文件 系统 缓冲 区 高 速 缓存 ,而 该 文件 系统 缓冲 区 高 速 缓存 正好 锁 住 了 ,那么 CPU 
可 以 决定 切换 至 另外 一 个 线程 而 不 是 等 待 。 有 关 是 进行 自 旋 还 是 进行 线程 切换 的 问题 则 是 
许多 研究 课题 的 内 容 , 下 面 会 讨论 其 中 的 一 部 分 。 请 注意 ,这 类 问题 在 单 处 理 机 中 是 不 存在 
的 ,因为 没有 另 一 个 CPU 释放 锁 , 那 么 自 旋 就 没有 任何 意义 ,如 果 一 个 线程 试图 取得 锁 并 
且 失 败 , 那 么 它 总 是 被 阻塞 ,这 样 的 锁 所 有 者 有 机 会 运行 和 释放 该 锁 。 
段 设 自 旋 和 进行 线程 切换 都 是 可 行 的 选择 , 则 可 进行 如 下 的 权衡 。 自 旋 直 接 浪费 了 
CPU 周期 。 重 复 地 测试 锁 并 不 是 高 效 的 工作 。 不 过 ,切换 也 浪费 了 CPU 周期 ,因为 必须 保 
存 当 前 线程 的 状态 ,必须 获得 保护 就 绪 链 表 的 锁 , 还 必须 选择 一 个 线程 ,必须 装 入 其 状态 ,并 
且 使 其 开始 运行 。 更 进一步 来 说 ,该 CPU 高 速 缓存 还 将 包含 所 有 不 合适 的 高 速 缓存 块 , 因 
此 在 线程 开始 运行 的 时 候 会 发 生 很 多 代价 昂贵 的 高 速 缓存 未 命中 。TLB 的 失效 也 是 可 能 
的 。 最 后 ,会 发 生 返 回 至 原来 线程 的 切换 , 随 之 而 来 的 是 更 多 的 高 速 缓存 未 命中 。 花 费 在 这 
两 个 线程 间 来 回 切换 和 所 有 高 速 缓 存 未 命中 的 周期 时 间 都 浪费 了 。 
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如 果 预 先知 道 互 斥 信号 量 通常 被 持 有 的 时 间 , 比 如 是 50ps, 而 从 当前 线程 切换 需要 
lms, 稍 后 切换 返回 还 需 lms', 那 么 在 互 斥 信号 量 上 自 旋 则 更 为 有 效 。 另 一 方面 ,如 果 互 斥 
信号 量 的 平均 保持 时 间 是 10ms, 那 就 值得 忍受 线程 切换 的 麻烦 。 问 题 在 于 ,临界 区 在 这 个 
期 间 会 发 生 相当 大 变化 ,所 以 哪 一 种 方法 更 好 些 昵 ? 

有 一 种 设计 是 总 是 进行 自 旋 。 第 二 种 设计 方案 则 总 是 进行 切换 。 而 第 三 种 设计 方案 是 
每 当 过 到 一 个 锁 住 的 互 斥 信 号 量 时 ,就 单独 做 出 决定 。 在 必须 做 出 决定 的 时 刻 ,并 不 知道 自 
旋 和 切换 哪 一 种 方案 更 好 ,但 是 对 于 任何 给 定 的 系统 ,有 可 能 对 其 所 有 的 有 关 活 动 进行 跟 
踪 , 并 且 随 后 进行 离线 分 析 。 然 后 就 可 以 确定 哪个 决定 最 好 及 在 最 好 的 情形 下 所 浪费 的 时 
间 。 这 种 事后 算法 成 为 对 可 行 算 法 进行 测量 的 基准 评测 标准 。 

已 有 研究 人 员 对 上 述 这 一 问题 进行 了 研究 。 多 数 的 研究 工作 使 用 了 这 样 一 个 模型 : 一 
个 未 能 获得 互 斥 信号 量 的 线程 自 旋 一 段 时 间 , 如 果 时 间 超 过 某 个 阔 值 , 则 进行 切换 。 在 某 些 
情形 下 ,该 六 值 是 一 个 定 值 ,典型 值 是 切换 至 另 一 个 线程 再 切换 回来 的 开销 。 在 另 一 些 情 形 
下 ,该 阔 值 是 动态 变化 的 , 它 取决 于 所 观察 到 的 等 待 互 斥 信号 量 的 历史 信息 。 

在 系统 跟踪 若干 最 新 的 自 旋 时 间 并 且 假 定 当前 的 情形 可 能 会 同 先前 的 情形 类 似 时 ,就 
可 以 得 到 最 好 的 结果 。 例 如 ,假定 还 是 lms 切换 时 间 , 线 程 自 旋 时 间 最 长 为 2ms, 但 是 要 观 
察 实际 上 自 旋 了 多 长 时 间 , 如 果 线 程 未 能 获取 锁 ,并 且 发 现在 之 前 的 三 轮 中 ,平均 等 待 时 间 
为 200ps, 那 么 在 切换 之 前 就 应 该 先 自 旋 2ms。 但 是 ,如 果 发 现在 先前 的 每 次 尝试 中 ,线程 
都 自 旋 了 整整 2ms, 则 应 该 立即 切换 而 不 再 自 旋 。 


9.6 多 处 理 机 调度 


在 探讨 多 处 理 机 调度 之 前 ,需要 确定 调度 的 对 象 是 什么 。 过 去 , 当 所 有 进程 都 是 单个 线 
程 的 时 候 , 调 度 的 单位 是 进程 ,因为 没有 其 他 什么 可 以 调度 的 。 所 有 的 现代 操作 系统 都 支持 
多 线程 进程 ,这 让 调度 变 得 更 加 复杂 。 

线程 是 内 核 线程 还 是 用 户 线程 至 关 重 要 。 如 果 线 程 是 由 用 户 空间 库 维 护 的 ,而 对 内 核 
不 可 见 , 那 么 调度 一 如 既往 地 基于 单个 进程 。 如 果 内 核 并 不 知道 线程 的 存在 , 它 就 不 能 调度 
线程 。 

对 内 核 线程 来 说 ,情况 有 所 不 同 。 在 这 种 情况 下 所 有 线程 均 是 内 核 可 见 的 ,内 核 可 以 选 
择 一 个 进程 的 任 一 线程 。 在 这 样 的 系统 中 ,发 展 趋势 是 内 核 选择 线程 作为 调度 单位 ,线程 从 
属 的 那个 进程 对 于 调度 算法 只 有 很 少 的 (乃至 没有 ) 影 响 。 下 面 将 探讨 线程 调度 ,当然 ,对 于 
一 个 单线 程 进 程 系统 或 者 用 户 空 间 线程 ,调度 单位 依然 是 进程 。 

进程 和 线程 的 选择 并 不 是 调度 中 的 唯一 问题 。 在 单 处 理 机 中 ,调度 是 一 维 的 。 唯 一 必 
须 (不 断 重 复 地 ) 回 答 的 问题 是 :“ 接 下 来 运行 的 线程 应 该 是 哪 一 个 ”而 在 多 处 理 机 中 ,调度 
是 二 维 的 。 调 度 程序 必须 决定 哪 一 个 进程 运行 以 及 在 哪 一 个 CPU 上 运行 。 这 个 在 多 处 理 
机 中 增加 的 维 数 大 大 增加 了 调度 的 复杂 性 。 

另 一 个 造成 复杂 性 的 因素 是 ,在 有 些 系 统 中 所 有 的 线程 是 不 相关 的 ,而 在 另外 一 些 系 统 
中 它们 是 成 组 的 ,同属 于 同一 个 应 用 并 且 协 同 工 作 。 前 一 种 情形 的 例子 是 分 时 系统 ,其 中 独 
立 的 用 户 运 行 相互 独立 的 进程 。 这 些 不 同 进程 的 线程 之 间 没 有 关系 ,因此 其 中 的 每 一 个 都 
可 以 独立 调度 而 不 用 考虑 其 他 的 线程 。 


后 一 种 情形 的 例子 通常 发 生 在 程序 开发 环境 中 ,大 型 系统 中 通常 有 一 些 供 实际 代码 使 
的 包含 宏 、 类 型 定义 以 及 变量 声明 等 内 容 的 头 文件 。 当 一 个 头 文件 改变 时 ,所 有 包含 它 的 
代码 文件 必须 被 重新 编译 。 通 常 make 程序 用 于 管理 开发 工作 。 调 用 make 程序 时 ,在 考虑 
了 头 文件 或 代码 文件 的 修改 之 后 , 它 仅 编译 那些 必须 重新 编译 的 代码 文件 。 仍 然 有 效 的 目 
标 文 件 不 再 重新 生成 。 

make 的 原始 版 本 是 顺序 工作 的 ,不 过 为 多 处 理 机 设计 的 新 版 本 可 以 一 次 启动 所 有 的 编 
译 。 如 果 需 要 10 个 编译 ,那么 迅速 对 9 个 进行 调度 而 让 最 后 一 个 在 很 长 时 间 之 后 才 进 行 的 
做 法 没有 多 大 意义 ,因为 直到 最 后 一 个 线程 完毕 之 后 用 户 才 感觉 到 工作 完成 了 。 在 这 种 情 
况 下 ,将 进行 编译 的 线程 看 作 一 组 ,并 在 对 其 调度 时 考虑 到 这 一 点 是 有 意义 的 。 

1. 分 时 

首先 讨论 调度 独立 线程 的 情况 , 稍 后 将 考虑 如 何 调度 相关 的 线程 。 处 理 独 立 线程 的 最 
简单 算法 是 ,为 就 绪 线程 维护 一 个 系统 级 的 数据 结构 , 它 可 能 只 是 一 个 链表 ,但 更 多 的 情况 
下 可 能 是 对 应 不 同 优先 级 的 一 个 链表 集合 ,如 图 9-14(a) 所 示 。 这 里 16 个 CPU 正在 忙碌 ， 
有 不 同 优先 级 的 12 个 线程 在 等 待 运行 。 第 一 个 将 要 完成 其 当前 工作 (或 其 线程 将 被 阻塞 ) 
的 CPU 是 CPU3, 然 后 CPU3 锁 住 调度 队列 并 选择 优先 级 最 高 的 线程 A, 如 图 9-14(b) 所 
示 。 接 着 ,CPU11 空闲 并 选择 线程 B, 参 见 图 9-14(c)。 只 要 线程 完全 无 关 , 以 这 种 方式 调 
度 是 明智 的 选择 并 且 其 很 容易 高 效 地 实现 。 
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图 9-14 使 用 单一 数据 结构 调度 多 处 理 机 


由 所 有 CPU 使 用 的 单个 调度 数据 结构 分 时 共享 这 些 CPU ,正如 它们 在 一 个 单 处 理 机 
系统 中 那样 。 它 还 支持 自动 负载 平衡 ,因为 绝 不 会 出 现 一 个 CPU 空闲 而 其 他 CPU 过 载 的 
情况 。 不 过 这 一 方法 有 两 个 缺点 ,一 个 是 随 着 CPU 数量 增加 所 引起 的 对 调度 数据 结构 的 
潜在 竞争 ,二 是 当 线程 由 于 IO 阻塞 时 所 引起 上 下 文 切 换 的 开销 。 

在 线程 的 时 间 片 用 完 时 ,也 可 能 发 生 上 下 文 切换 。 在 多 处 理 机 中 它 有 一 些 在 单 处 理 机 
中 不 存在 的 属性 。 假 设 某 个 线程 在 其 时 间 片 用 完 时 持 有 一 把 自 旋 锁 。 在 该 线程 被 再 次 调度 
并 且 释 放 该 锁 之 前 ,其 他 等 待 该 自 旋 锁 的 CPU 只 是 把 时 间 浪 费 在 自 旋 上 。 在 单 处 理 机 中 ， 
极 少 采用 自 旋 锁 , 因 此 ,如 果 持 有 互 斥 信号 量 的 一 个 线程 被 挂 起 ,而 另 一 个 线程 启动 并 试图 
获取 该 互 斥 信号 量 , 则 该 线程 会 立即 被 阻塞 ,这 样 只 浪费 了 少量 时 间 。 
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为 了 避免 这 种 异常 情况 ,一些 系 统 采用 智能 调度 的 方法 ,其 中 ,获得 了 自 旋 锁 的 线程 设 
置 一 个 进程 范围 内 的 标志 以 表示 它 目 前 拥有 了 一 个 自 旋 锁 。 当 它 释 放 该 自 旋 锁 时 ,就 清除 
这 个 标志 。 这 样 调度 程序 就 不 会 停止 持 有 自 旋 锁 的 线程 ,相反 ,调度 程序 会 给 予 稍微 多 一 些 
的 时 间 让 该 线程 完成 临界 区 的 工作 并 释放 自 旋 锁 。 

调度 中 的 另 一 个 主要 问题 是 , 当 所 有 CPU 平等 时 , 某 些 CPU 更 平等 。 特 别 是 , 当 线 程 
A 已 经 在 CPUs 上 运行 了 很 长 一 段 时 间 时 ,CPU 的 高 速 缓存 装 满 了 A 的 块 。 若 A 很 快 重 
新 开始 运行 ,那么 如 果 它 在 CPUs 上 运行 性 能 可 能 会 更 好 一 些 ,因为 KK 的 高 速 缓存 也 许 还 
存 有 A 的 一 些 块 。 预 装 高 速 缓存 块 将 提高 高 速 缓存 的 命中 率 ,从 而 提高 了 线程 的 速度 。 另 
外 ,TLB 也 可 能 含有 正确 的 页 面 ,从 而 减少 了 TLB 失效 。 

有 些 多 处 理 机 考虑 了 这 一 因素 ,并 使 用 了 亲 和 调 度 。 其 基本 思想 是 ,尽量 使 一 个 线程 在 
它 前 一 次 运行 过 的 同一 个 CPU 上 和 运行。 创建 这 种 亲和力 的 一 种 途径 是 采用 一 种 两 级 调度 
算法 。 在 一 个 线程 创建 时 , 它 被 分 给 一 个 CPU ,例如 ,可 以 基于 哪 一 个 CPU 在 此 刻 有 最 小 
的 负载 。 这 种 把 线程 分 给 CPU 的 工作 在 算法 的 顶层 进行 ,其 结果 是 每 个 CPU 获得 了 自己 
的 线程 集 。 

线程 的 实际 调度 工作 在 算法 的 底层 进行 。 它 由 每 个 CPU 使 用 优先 级 或 其 他 的 手段 分 
别 进行 。 通 过 试图 让 一 个 线程 在 其 生命 周期 内 在 同一 个 CPU 上 运行 的 方法 ,高 速 缓存 的 
亲和力 得 到 了 最 大 化 。 不 过 ,如 果 某 一 个 CPU 没有 线程 运行 , 它 便 选取 另 一 个 CPU 的 一 
个 线程 来 运行 而 不 是 空转 。 

两 级 调度 算法 有 三 个 优点 。 第 一 , 它 把 负载 大 致 平均 地 分 配 在 可 用 的 CPU 上; 第 二 ， 
它 尽 可 能 发 挥 了 高 速 缓存 亲和力 的 优势 ; 第 三 ,通过 为 每 个 CPU 提供 一 个 私有 的 就 绪 线程 
链表 ,使 得 对 就 绪 线程 链表 的 竞争 减 到 了 最 小 ,因为 试图 使 用 另 一 个 CPU 的 就 绪 线程 链表 
的 机 会 相对 较 小 。 

2. 空间 共享 

当 线程 之 间 以 某 种 方式 彼此 相关 时 ,可 以 使 用 其 他 多 处 理 机 调度 方法 。 经 常 有 一 个 进 
程 有 多 个 共同 工作 的 线程 的 情况 发 生 。 例 如 , 当 一 个 进程 的 多 个 线程 间 频 繁 地 进行 通信 ,让 
其 在 同一 时 间 执 行 就 显得 尤为 重要 。 在 多 个 CPU 上 同时 调度 多 个 线程 称 为 空间 共享 。 

最 简单 的 空间 共享 算法 是 这 样 工作 的 。 假 设 一 组 相关 的 线程 是 一 次 性 创建 的 。 在 其 创 
建 的 时 刻 ,调度 程序 检查 是 否 有 同 线程 数量 一 样 多 的 空闲 CPU 存在 。 如 果 有 ,每 个 线程 获 
得 各 自 专用 的 CPU( 非 多 道 程序 处 理 ) 并 且 都 开始 运行 。 如 果 没 有 足够 的 CPU ,就 没有 线 
程 开 始 运 行 ,直到 有 足够 的 CPU 时 为 止 。 每 个 线程 保持 其 CPU 直到 它 终止 ,并 且 该 CPU 
被 送 回 可 用 CPU 池 中 。 如 果 一 个 线程 在 IO 上 阻塞 , 它 继续 保持 其 CPU, 而 该 CPU 就 空 
闲 直 到 该 线程 被 唤醒 。 在 下 一 批 线程 出 现时 ,应 用 同样 的 算法 。 

在 任何 一 个 时 刻 , 全 部 CPU 被 静态 地 划分 成 若干 个 分 区 ,每 个 分 区 都 运行 一 个 进程 中 
的 线程 。 例 如 ,在 图 9-15 中 ,分 区 的 大 小 是 4.6、8 和 12 个 CPU, 有 两 个 CPU 没有 分 配 。 随 
着 时 间 的 流逝 ,新 的 线程 创建 , 旧 的 线程 终止 ,CPU 分 区 大 小 和 数量 都 会 发 生变 化 。 

必须 进行 周期 性 的 调度 决策 。 在 单 处 理 机 系统 中 ,最 短 作业 优先 是 批 处 理 调度 中 知名 
的 算法 。 在 多 处 理 机 系统 中 类 似 的 算法 是 ,选择 需要 最 少 的 CPU 周期 数 的 线程 ,也 就 是 其 
CPU 周期 数 工 运行 时 间 最 小 的 线程 为 候选 线程 。 然 而 ,在 实际 中 ,这 一 信息 很 难得 到 ,因此 
该 算法 难以 实现 。 事 实 上 ,研究 表明 ,要 胜 过 先 来 先 服 务 算法 是 非常 困难 的 。 
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图 9-15 32 个 CPU 的 集合 被 分 成 4 个 分 区 ,两 个 CPU 可 用 


在 这 个 简单 的 分 区 模型 中 ,一 个 线程 请 求 一 定数 量 的 CPU, 然 后 或 者 全 部 得 到 它们 或 
者 一 直 等 到 有 足够 数量 的 CPU 可 用 为 止 。 另 一 种 处 理 方式 是 主动 地 管理 线程 的 并 行 度 。 
管理 并 行 度 的 一 种 途径 是 使 用 一 个 中 心服 务 器 ,用 它 跟 踪 哪些 线程 正在 运行 ,哪些 线程 希望 
运行 以 及 所 需 CPU 的 最 小 和 最 大 数量 。 每 个 应 用 程序 周期 性 地 询问 中 心服 务 器 有 多 少 个 
CPU 可 用 。 然 后 它 调整 线程 的 数量 以 符合 可 用 的 数量 。 例 如 ,一 台 Web 服务 器 可 以 有 5、 
10、20 或 任何 其 他 数量 的 线程 并 行 运行 。 如 果 它 当前 有 10 个 线程 ,突然 ,系统 对 CPU 的 需 
求 增加 了 ,于 是 它 被 通知 可 用 的 CPU 数量 减 到 了 5 个 ,那么 在 接 下 来 的 5 个 线程 完成 其 当 
前 工作 之 后 ,它们 就 被 通知 退出 而 不 是 给 予 新 的 工作 。 这 种 机 制 允 许 分 区 大 小 动态 地 变化 ， 
以 便 与 当前 负载 相 匹 配 ,这 种 方法 优 于 图 9-15 中 的 固定 系统 。 

3， 和 群 调度 

空间 共享 的 一 个 明显 优点 是 消除 了 多 道 程序 设计 ,从 而 消除 了 上 下 文 切 换 的 开销 。 但 
是 ,一 个 同样 明显 的 缺点 是 当 CPU 被 阻塞 或 根本 无 事 可 做 时 时 间 被 浪费 了 ,只 有 等 到 其 再 
次 就 绪 。 于 是 ,人 们 寻找 既 可 以 调度 时 间 又 可 以 调度 空间 的 算法 ,特别 是 对 于 要 创建 多 个 线 
程 而 这 些 线程 通常 需要 彼此 通信 的 线程 。 

为 了 考察 一 个 进程 的 多 个 线程 被 独立 调度 时 会 出 现 的 问题 ,设想 一 个 系统 中 有 线程 Au 
和 Al 属于 进程 A, 而 线程 B 和 Bi 属于 进程 B。 线程 A 和 Bo 在 CPU0 上 分 时 ,而 线程 Ai 
和 Bi 在 CPU1 上 分 时 。 线 程 A 和 Ai 需要 经 常 通信 。 其 通信 模式 是 , A。 送 给 Al 一 个 消 
息 ,然后 Ai 回 送 给 Au 一 个 应 答 , 紧 跟 的 是 另 一 个 这 样 的 序列 。 假 设 正好 是 Av 和 Bi 首先 
开始 ,如 图 9-16 所 示 。 
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时 间 0 100 200 300 400 500 600 


图 9-16 进程 A 的 两 个 异步 运行 的 线程 间 的 通信 


在 时 间 片 0,Ao 发 给 Ai 一 个 请 求 ,但 是 直到 Ai 在 开始 于 100ms 的 时 间 片 1 中 开始 运 
行 时 它 才 得 到 该 消息 。 它 立即 发 送 一 个 应 答 , 但 是 直到 Au 在 200ms 再 次 运行 时 它 才 得 到 
该 应 答 。 最 终结 果 是 每 200ms 一 个 请 求 -应 答 序列 。 这 个 结果 并 不 好 。 

这 一 问题 的 解决 方案 是 群 调度 , 它 是 协同 调度 的 发 展 产 物 。 群 调度 由 以 下 三 个 部 分 
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组 成 。 
(1) 把 一 组 相关 线程 作为 一 个 单位 , 即 一 个 群 ,一 起 调度 。 
(2) 一 个 群 中 的 所 有 成 员 在 不 同 的 分 时 CPU 上 同时 运行 。 


(3) 群 中 的 所 有 成 员 共 同 开始 和 结束 其 时 间 片 。 

使 群 调度 正确 工作 的 关键 是 ,同步 调度 所 有 的 CPU。 这 意味 着 把 时 间 划 分 为 离散 的 时 
间 片 ,如 图 9-16 所 示 。 在 每 一 个 新 的 时 间 片 开始 时 ,所 有 的 CPU 都 重新 调度 ,在 每 个 CPU 
上 都 开始 一 个 新 的 线程 。 在 后 续 的 时 间 片 开始 时 , 另 一 个 调度 事件 发 生 。 在 这 之 间 , 没 有 调 
度 行为 。 如 果 某 个 线程 被 阻塞 , 它 的 CPU 保持 空闲 ,直到 对 应 的 时 间 片 结束 为 止 。 

有 关 群 调度 是 如 何 工作 的 例子 在 图 9-17 中 给 出 。 图 9-17 中 有 一 台 带 6 个 CPU 的 多 
处 理 机 ,由 5 个 进程 A~E 使 用 ,总 共有 24 个 就 绪 线程 。 在 时 间 槽 0, 线 程 Au 一 As 被 调度 
运行 。 在 时 间 槽 1 ,调度 线程 B.B, 、Bs .Co .Ci 和 Cs 被 调度 运行 。 在 时 间 槽 2, 进 程 D 的 5 
个 线程 以 及 E。 运行 。 剩 下 的 6 个 线程 属于 下 ,在 时 间 槽 3 中 运行 。 然 后 周期 重复 进行 ,时 
间 槽 4 与 时 间 槽 0 一 样 ,以 此 类 推 。 
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图 9-17 群 调度 


群 调度 的 思想 是 ,让 一 个 进程 的 所 有 线程 一 起 运行 ,这 样 , 如 果 其 中 一 个 线程 向 另 一 个 
线程 发 送 请 求 ,接收 方 几乎 会 立即 得 到 消息 ,并 且 几 乎 能 够 立即 应 答 。 在 图 9-17 中 ,由 于 进 
程 的 所 有 线程 在 同一 个 时 间 片 内 一 起 运行 ,它们 可 以 在 一 个 时 间 片 内 发 送 和 接收 大 量 的 消 
息 , 从 而 消除 了 图 9-16 中 的 问题 。 


9.7 多 处 理 器 、 超 线程 和 多 核 的 比较 


多 处 理 器 、 超 线程 和 多 核 的 共同 点 是 均 为 了 提升 计算 机 性 能 而 设计 , 均 可 以 同时 执行 多 
个 指令 序列 。 但 是 区 别 也 很 明显 。 主 要 体现 在 同时 执行 的 两 个 线程 之 间 共 享 物理 资源 的 多 
少 。 多 处 理 器 的 共享 物理 资源 最 少 ,每 个 线程 有 自己 单独 的 处 理 器 : 超 线程 共享 最 多 ， 
ALU、FPU、MSR 缓存 等 均 为 共享 物理 资源 ; 而 多 核 则 介 于 两 者 之 间 , 共 享 处 理 器 ,但 不 共 
享 ALU FPU 等 。 具 体 来 说 ,我 们 有 

对 于 HT 线程 来 说 ,其 共享 的 资源 包括 ALU、 某 些 MSR 和 缓存 ; 而 其 独 享 的 资源 有 本 
地 APIC、 通 用 寄存 器 .Ll 缓存 .CPUID 等 。 

对 于 多 核 来 说 ,共享 资源 包括 最 后 一 级 的 缓存 (如 英特尔 的 智能 缓存 (Intel Smart 


Cache))、 一 少 部 分 寄存 器 如 MSR, 能 供 管理 单元 也 有 可 能 共享 。 而 独 享 的 有 CPUID、 
APIC、BIOS 等 。 

因为 共享 资源 数量 不 同 ,多 处 理 器 、 超 线程 、 多 核 的 成 本 自然 也 不 相同 。 多 处 理 器 成 本 
最 高 ,独立 性 最 高 、 功 耗 最 大 ; 超 线程 的 成 本 、 独 立 性 和 功 耗 最 小 ; 而 多 核 则 处 于 中 间 。 

这 里 值得 一 提 的 是 , 超 线程 技术 是 英特尔 公司 所 独 有 的 ,其 他 公司 不 一 定 使 用 这 种 技 
术 。 例 如 ,AMD 公司 就 直接 从 多 处 理 器 跨越 到 了 多 核 。 


小 结 


采用 多 个 CPU 可 以 把 计算 机 系统 建造 得 更 快 更 可 靠 。 在 x86 体系 结构 下 ,多 处 理 功 能 
芯片 经 过 了 对 称 多 处 理 器 结构 、 超 线程 结构 多核 结 构 和 多 核 超 线程 结构 的 4 个 演变 阶段 。 
各 种 操作 系统 的 配置 都 是 可 能 的 ,包括 给 每 个 CPU 配 一 个 各 自 的 操作 系统 ,配置 一 个 主 操 
作 系统 而 其 他 是 从 属 的 操作 系统 ,或 者 是 一 个 对 称 多 处 理 器 ,在 每 个 CPU 上 都 可 以 运行 操 
作 系 统 的 一 个 副本 。 在 后 一 种 情形 下 ,需要 用 锁 提 供 同步 。 当 没有 可 用 的 锁 时 ,一 个 CPU 
会 空转 或 者 进行 上 下 文 切换 。 各 种 调度 算法 都 是 可 能 的 ,包括 分 时 、 空 间 分 割 以 及 群 调 度 。 
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也 请 您 发 邮件 告诉 我 们 ,以 便 我 们 更 好 地 为 您 服务 。 


我 们 的 联系 方式 : 
地 址 ; 北京 海淀 区 双 清 路 学 研 大 厦 A 座 707 
邮 编 : 100084 


电 话 : 010 一 62770175 一 4604 


资源 下 载 : http://www. tup. com. cn 


所 二 本 
资源 下 载 、 样 书 申 请 
新 书 推荐 .技术 交流 


电子 邮件 : weij@tup.tsinghua. edu. cn 
QQ: 883604( 请 写 明 您 的 单位 和 姓名 ) 


用 微 信 扫 一 扫 右 边 的 二 维 码 , 即 可 关注 清华 大 学 出 版 社 公 众 号 “ 书 圈 ”。 


